Compare commits

...

375 Commits

Author SHA1 Message Date
Jonas Bark
6ea4fa82a7 version++ 2025-12-20 11:39:02 +01:00
Jonas Bark
0d78ca6352 version++ 2025-12-20 11:20:39 +01:00
Jonas Bark
b82ad80d1c fix windows build version 2025-12-20 10:56:19 +01:00
Jonas Bark
eece2ce7dc update changelog 2025-12-20 10:30:32 +01:00
jonasbark
3bfd7dd7ab Merge pull request #222 from jonasbark/copilot/implement-in-app-purchase
Implement IAP system with 5-day trial and command limiting
2025-12-20 10:13:33 +01:00
Jonas Bark
b0f3dffc3c screenshots 2025-12-20 10:13:20 +01:00
Jonas Bark
f5234d0c11 update changelog 2025-12-20 10:06:57 +01:00
Jonas Bark
8a12ccb01e cleanup 2025-12-20 10:04:44 +01:00
Jonas Bark
fac2e86240 bugfixes and clarifications 2025-12-20 10:01:37 +01:00
Jonas Bark
39b49bb9de sram fix 2025-12-19 21:06:13 +01:00
Jonas Bark
406ccfa2ce adjust iOS receipt logic 2025-12-19 20:41:45 +01:00
Jonas Bark
16e6b96cc7 SRAM AXS support 2025-12-19 20:41:27 +01:00
Jonas Bark
1e37c8a742 press longer for the simulator, enable long press for devices not supporting long press (e.g. Shimano Di2) 2025-12-19 11:40:07 +01:00
Jonas Bark
5f0389b36f press longer for the simulator, enable long press for devices not supporting long press (e.g. Shimano Di2) 2025-12-19 11:38:30 +01:00
Jonas Bark
8762d85d57 press longer for the simulator, enable long press for devices not supporting long press (e.g. Shimano Di2) 2025-12-19 11:33:13 +01:00
Jonas Bark
018bbd43f1 fix 2025-12-19 10:13:01 +01:00
Jonas Bark
756e3fc556 version++ 2025-12-19 10:09:06 +01:00
Jonas Bark
7ac5f5dbcb fine tune gryoscope logic to react more quickly 2025-12-19 09:29:19 +01:00
Jonas Bark
65c613fd24 fine tune gryoscope logic to react more quickly 2025-12-19 09:29:05 +01:00
Jonas Bark
e0ad9007a4 ui adjustments 2025-12-19 09:21:11 +01:00
Jonas Bark
c3cda5f547 ui adjustments, add polish translations 2025-12-19 08:56:14 +01:00
Jonas Bark
12f379c03d screenshot fix 2025-12-18 15:39:16 +01:00
Jonas Bark
e760c34ede unit test fixes 2025-12-18 15:30:14 +01:00
Jonas Bark
187a15a55d improve trainer controller view 2025-12-18 14:51:49 +01:00
Jonas Bark
3e4751a6b5 refactor sensor logic 2025-12-18 12:24:52 +01:00
Jonas Bark
b05d87196a implement functionality, refactoring 2025-12-18 12:06:34 +01:00
Jonas Bark
b8297848f6 Merge branch 'copilot/add-steering-detection-algorithm' into copilot/implement-in-app-purchase 2025-12-18 11:37:39 +01:00
Jonas Bark
de0f004a48 implement functionality, refactoring 2025-12-18 11:37:21 +01:00
Jonas Bark
0bc9c1d4d2 implement functionality, refactoring 2025-12-18 11:33:14 +01:00
Jonas Bark
3fccb59544 Merge branch 'copilot/implement-in-app-purchase' into copilot/add-steering-detection-algorithm 2025-12-18 09:46:45 +01:00
Jonas Bark
ea76aabf91 update changelog 2025-12-18 09:46:32 +01:00
Jonas Bark
dc1c900bd2 bugfix and animation 2025-12-18 09:26:25 +01:00
Jonas Bark
b1e59d8f2a auto-open key editor on key press 2025-12-18 08:46:31 +01:00
Jonas Bark
006148dbd0 move button simulator page button 2025-12-18 08:33:04 +01:00
Jonas Bark
974ba258f5 teaser new connection methods 2025-12-18 08:11:06 +01:00
copilot-swe-agent[bot]
29d833e9d1 Update README with gyroscope steering device information
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-18 06:55:33 +00:00
copilot-swe-agent[bot]
eea72405bb Fix code review issues in gyroscope steering implementation
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-18 06:54:42 +00:00
copilot-swe-agent[bot]
e500f1ed0b Add unit tests for gyroscope steering algorithm
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-18 06:52:39 +00:00
copilot-swe-agent[bot]
6c3bd2e6a1 Add gyroscope/accelerometer steering support with sensor fusion
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-18 06:51:43 +00:00
copilot-swe-agent[bot]
e54d8968e8 Initial plan 2025-12-18 06:44:46 +00:00
Jonas Bark
9dca2e989a check sandbox on iOS / macOS 2025-12-17 15:57:44 +01:00
Jonas Bark
610c5d6ef5 check sandbox on iOS / macOS 2025-12-17 15:48:30 +01:00
Jonas Bark
7797b34852 improve keymap UI 2025-12-17 14:34:08 +01:00
Jonas Bark
99e9f326f7 fix build 2025-12-17 13:05:05 +01:00
Jonas Bark
0f2d73239b fix build 2025-12-17 12:57:32 +01:00
Jonas Bark
497b489ea9 remove Podfile.loc 2025-12-17 12:40:29 +01:00
Jonas Bark
51581f106a cleanup README.md 2025-12-17 12:17:00 +01:00
Jonas Bark
43e9aa02e0 cleanup README.md 2025-12-17 12:13:23 +01:00
Jonas Bark
089a41cc2b translations and ux changes 2025-12-17 11:39:39 +01:00
Jonas Bark
279ab101cc macOS fix 2025-12-17 11:13:42 +01:00
Jonas Bark
c0f278652e show connection update notifications 2025-12-17 10:54:16 +01:00
Jonas Bark
ac02cc78bc show connection update notifications 2025-12-17 10:34:40 +01:00
Jonas Bark
d42ac3af6b notifications on iOS 2025-12-17 10:20:36 +01:00
Jonas Bark
20cfe76091 limit to 80 on Android during trial period, only count sent out commands 2025-12-17 10:02:32 +01:00
Jonas Bark
b68170b489 fix purchause logic 2025-12-17 09:08:28 +01:00
Jonas Bark
21273fa9ca Merge branch 'main' into copilot/implement-in-app-purchase
# Conflicts:
#	CHANGELOG.md
#	lib/bluetooth/devices/base_device.dart
#	pubspec.yaml
2025-12-17 08:41:47 +01:00
jonasbark
700afb1050 Document local connection method for Rouvy
Added instructions for the local connection method in Rouvy.
2025-12-16 19:56:56 +01:00
Jonas Bark
d943544e56 Merge branch 'main' of github.com:jonasbark/swiftcontrol 2025-12-16 19:53:42 +01:00
Jonas Bark
e994eb01dd less confusing messages 2025-12-16 19:53:25 +01:00
jonasbark
59d953bbc4 Update Windows Store version to 4.1.0 2025-12-16 14:36:30 +01:00
Jonas Bark
12ecbf80e1 version++ 2025-12-16 12:19:03 +01:00
Jonas Bark
a5b76b43bf build fix 2025-12-16 11:25:25 +01:00
Jonas Bark
383055bfe2 flutter version 2025-12-16 10:57:28 +01:00
Jonas Bark
24212e8e4c update changelog 2025-12-16 10:50:36 +01:00
Jonas Bark
1513a53dd4 update MyWhoosh keymap to use A and D keyboard keys for steering 2025-12-16 10:41:10 +01:00
Jonas Bark
fe9dd29964 cleanup, translations 2025-12-16 10:14:48 +01:00
Jonas Bark
eece2bcc0f Merge branch 'main' into copilot/implement-in-app-purchase
# Conflicts:
#	lib/bluetooth/devices/openbikecontrol/obc_ble_emulator.dart
2025-12-16 09:19:50 +01:00
Jonas Bark
3ca63c0523 permission order 2025-12-16 09:19:06 +01:00
Jonas Bark
b46982918d resolve #224 2025-12-15 20:16:30 +01:00
Jonas Bark
13b075a1f3 only show supported actions 2025-12-15 13:48:18 +01:00
Jonas Bark
f71a417ac5 only show supported actions 2025-12-15 13:47:43 +01:00
Jonas Bark
27065f2906 prefill obp keymap actions 2025-12-15 13:42:54 +01:00
Jonas Bark
b5d938fd47 error reporting when starting connection method 2025-12-15 13:13:57 +01:00
Jonas Bark
46300fc0d4 hide other connection methods in an accordion 2025-12-15 13:03:37 +01:00
Jonas Bark
93882b8b36 prioritize OBP connection methods when available 2025-12-15 12:48:19 +01:00
jonas.bark@gmx.de
f19c6b8dd0 fix windows implementation 2025-12-15 11:12:06 +01:00
Jonas Bark
968e2c5928 fix button mapping for OpenBikeControl, button simulator changes 2025-12-15 09:49:07 +01:00
Jonas Bark
44599b2d33 fix button mapping for OpenBikeControl, button simulator changes 2025-12-15 09:48:36 +01:00
Jonas Bark
613f75fd25 Merge branch 'main' into copilot/implement-in-app-purchase 2025-12-15 09:22:18 +01:00
Jonas Bark
c09ab5482e fix button mapping for OpenBikeControl 2025-12-15 09:13:44 +01:00
Jonas Bark
6f68e6cb62 iap adjustment 2025-12-15 08:37:56 +01:00
Jonas Bark
43ac412efd iap adjustment 2025-12-14 21:35:06 +01:00
Jonas Bark
7149c98564 windows #2 2025-12-14 20:15:33 +01:00
Jonas Bark
0f4e46a758 windows #1 2025-12-14 19:54:41 +01:00
Jonas Bark
23fb927cd6 in app purchase implementation on macOS / iOS 2025-12-14 19:32:23 +01:00
Jonas Bark
d055a260ab fix shadcn design usage 2025-12-14 16:48:04 +01:00
Jonas Bark
d55fa5f7c0 Merge branch 'main' into copilot/implement-in-app-purchase
# Conflicts:
#	lib/bluetooth/devices/base_device.dart
#	lib/pages/configuration.dart
#	lib/utils/settings/settings.dart
2025-12-14 16:42:50 +01:00
Jonas Bark
5cc49bd246 no device information service on Windows 2025-12-14 16:31:14 +01:00
Jonas Bark
c3c49decd1 remove remains of swift_control 2025-12-14 16:22:07 +01:00
Jonas Bark
127c997ea1 less warnings for Click V2 users 2025-12-13 15:17:28 +01:00
Jonas Bark
724d52ba10 add keyboard input to supported devices 2025-12-13 10:22:55 +01:00
Jonas Bark
5cee0fbf55 use latest flutter 2025-12-13 10:11:28 +01:00
Jonas Bark
e44760d0e3 Revert "patch Android only"
This reverts commit 0eba068910.
2025-12-12 10:48:24 +01:00
Jonas Bark
29be3c4411 Merge remote-tracking branch 'origin/main' 2025-12-12 09:08:17 +01:00
Jonas Bark
0eba068910 patch Android only 2025-12-12 09:08:07 +01:00
Jonas Bark
04dee3f14c fix wrong permission for BLE advertising on Android 2025-12-12 09:06:33 +01:00
copilot-swe-agent[bot]
f4fd658c36 Refactor to reduce code duplication and use constants for magic numbers
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-12 07:47:41 +00:00
copilot-swe-agent[bot]
0e80d5612c Fix code review issues: remove unused import and fix negative counter display
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-12 07:45:29 +00:00
copilot-swe-agent[bot]
9302ebc667 Add restore purchases functionality to IAP widget
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-12 07:43:00 +00:00
copilot-swe-agent[bot]
2265866f58 Clarify existing user detection logic with comments
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-12 07:41:11 +00:00
copilot-swe-agent[bot]
8ec6ee5ef0 Improve IAP error handling and add documentation
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-12 07:39:13 +00:00
copilot-swe-agent[bot]
a03d250bdb Add IAP service implementation with trial and command limiting
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-12 07:37:15 +00:00
copilot-swe-agent[bot]
a0ebac41ea Initial plan 2025-12-12 07:29:07 +00:00
jonasbark
b4b3f5db67 Improve Click V2 connection instructions
Updated troubleshooting steps for Click V2 connection reliability.
2025-12-11 23:47:15 +01:00
Jonas Bark
b291f59e10 change icons 2025-12-11 22:55:54 +01:00
Jonas Bark
02de453952 Windows: fix version check URL 2025-12-11 22:53:16 +01:00
Jonas Bark
4c53f6e408 adjust windows logo 2025-12-11 22:27:29 +01:00
Jonas Bark
4f4d67cccc Merge remote-tracking branch 'origin/main' 2025-12-11 20:44:51 +01:00
Jonas Bark
ef056f0503 web log debugging 2025-12-11 20:44:41 +01:00
jonasbark
a6f5755b42 Update MyWhoosh Link instructions for clarity
Emphasized the importance of closing MyWhoosh Link and added clarification about app connectivity issues.
2025-12-11 19:38:02 +01:00
jonasbark
5ce6e37973 Add troubleshooting section for MyWhoosh clicks
Added troubleshooting tip for unrecognized clicks in MyWhoosh.
2025-12-11 18:27:40 +01:00
Jonas Bark
ebebd7ad8b version++ 2025-12-11 10:22:26 +01:00
Jonas Bark
f6c47e3dab fix markdown theme colors 2025-12-11 10:21:28 +01:00
jonasbark
de711e12dc Merge pull request #221 from jonasbark/copilot/configure-hotkeys-for-buttons
Add keyboard hotkey configuration for button simulator
2025-12-11 08:26:35 +00:00
Jonas Bark
b94fed2f21 add wahoo kickr support to readme 2025-12-11 09:26:08 +01:00
Jonas Bark
9316881048 button simulator adjustments 2025-12-11 09:20:02 +01:00
copilot-swe-agent[bot]
c60a990938 Add mounted check in delayed callback and length validation for keys
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-11 07:24:50 +00:00
copilot-swe-agent[bot]
e9aaa96185 Fix remaining code review issues: remove redundant check and await settings save
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-11 07:23:15 +00:00
copilot-swe-agent[bot]
cb497daee4 Address code review feedback: extract constants and fix async handling
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-11 07:21:57 +00:00
copilot-swe-agent[bot]
4881fe4778 Add test for button simulator hotkeys and fix trailing comma
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-11 07:19:28 +00:00
copilot-swe-agent[bot]
5d5d8ffb18 Add keyboard hotkey configuration for button simulator
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-11 07:17:31 +00:00
copilot-swe-agent[bot]
b707812a7e Initial plan 2025-12-11 07:12:08 +00:00
Jonas Bark
9de50dc6fc markdown changes 2025-12-11 08:09:50 +01:00
Jonas Bark
11d308b53b markdown changes 2025-12-11 08:07:12 +01:00
Jonas Bark
0e03ec4a03 clarify mywhoosh link 2025-12-10 21:26:43 +01:00
Jonas Bark
68a04fad96 handling fix 2025-12-10 21:21:23 +01:00
Jonas Bark
ce75fd0f34 add more instructions, clarify mail support 2025-12-10 21:17:49 +01:00
Jonas Bark
d46b71b2d0 md #1 2025-12-10 20:23:41 +01:00
Jonas Bark
6492afc46f MyWhoosh: updated default keymap to use steering instead of navigating 2025-12-10 19:14:13 +01:00
Jonas Bark
c9b068e1b3 control your trainer manually without requiring a controller - just like a Companion app 2025-12-10 18:57:08 +01:00
Jonas Bark
5cdf15a419 UI adjustments 2025-12-10 17:26:00 +01:00
Jonas Bark
24db720927 don't show firmware update warning when we don't have that info, yet 2025-12-10 09:16:57 +01:00
Jonas Bark
94754d3d9b Rouvy doesn't support network controllers, yet 2025-12-10 09:14:07 +01:00
Jonas Bark
bffdae1a9b enable local connection on Windows if the app doesn't support OBP 2025-12-10 09:08:54 +01:00
Jonas Bark
a8b68c2d89 disable MyWhoosh Link connection method when running BikeControl and MyWhoosh on Windows on the same device 2025-12-10 09:03:58 +01:00
Jonas Bark
84f70f13d8 Gamepads: handle analog values correctly on Windows 2025-12-10 08:58:41 +01:00
Jonas Bark
ef1048ec08 adjust Headwind logic according to comment from https://github.com/jonasbark/swiftcontrol/issues/11#issuecomment-3634041684 2025-12-09 21:56:45 +01:00
Jonas Bark
37bc2110f5 update screenshots 2025-12-09 17:59:10 +01:00
Jonas Bark
84fd828d36 work on issue #11 2025-12-09 09:00:01 +01:00
Jonas Bark
a51b4d7958 version++ 2025-12-08 20:15:41 +01:00
Jonas Bark
117467d708 fix issue #215 2025-12-08 19:11:03 +01:00
Jonas Bark
789509f9cf fix build 2025-12-08 18:01:47 +01:00
Jonas Bark
a323dc213d fix build 2025-12-08 18:00:02 +01:00
jonasbark
0ec998a618 Merge pull request #212 from jonasbark/copilot/add-headwind-support
Add KICKR Headwind fan control support
2025-12-08 16:50:23 +00:00
copilot-swe-agent[bot]
0f5e9d59a8 Implement proper Headwind protocol with state tracking and mode switching
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-08 16:48:56 +00:00
copilot-swe-agent[bot]
2280fda916 Use firstOrNull to avoid potential race condition
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-08 16:29:55 +00:00
copilot-swe-agent[bot]
8d4db788a3 Refactor Headwind control logic and remove buttons
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-08 16:27:47 +00:00
Jonas Bark
c2bfc472fe fix patch pipeline 2025-12-08 11:20:31 +01:00
Jonas Bark
09ffd258b7 version++ 2025-12-08 11:09:43 +01:00
Jonas Bark
4ae92ca557 Merge remote-tracking branch 'origin/main' 2025-12-08 11:08:04 +01:00
Jonas Bark
e066054681 fix Di2 buttons not triggering an event 2025-12-08 11:07:55 +01:00
Jonas Bark
a0f4aadd37 fix repeated clicks on analog buttons for gamepads 2025-12-08 10:48:15 +01:00
copilot-swe-agent[bot]
3566dbc37c Address code review feedback: improve type annotations and validation
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-08 08:26:29 +00:00
copilot-swe-agent[bot]
73a23e06ba Fix syntax issues in ButtonEditPage and add missing import
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-08 08:25:09 +00:00
copilot-swe-agent[bot]
a03576d415 Add KICKR Headwind support implementation
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-08 08:23:27 +00:00
copilot-swe-agent[bot]
079db14127 Initial plan 2025-12-08 08:16:38 +00:00
jonasbark
2f4764a01f Update Windows Store version to 4.0.0 2025-12-08 08:57:21 +01:00
Jonas Bark
2671a9807b cleanup 2025-12-07 14:41:06 +01:00
Jonas Bark
da46deb495 fix build 2025-12-07 13:03:54 +01:00
Jonas Bark
0700bd331f fix build 2025-12-07 12:41:14 +01:00
Jonas Bark
14676d9277 Merge branch 'feature/openbikecontrol' 2025-12-07 12:37:05 +01:00
Jonas Bark
3f9c3611ec fix build 2025-12-07 12:36:57 +01:00
jonasbark
26263f4d6b Merge pull request #208 from jonasbark/feature/openbikecontrol
4.0.0
2025-12-07 11:13:09 +00:00
Jonas Bark
105a644599 better web compatibility 2025-12-07 12:10:53 +01:00
Jonas Bark
ba54234734 better web compatibility 2025-12-07 12:05:52 +01:00
Jonas Bark
d39af2c8ff show keys instead of toast 2025-12-07 11:30:06 +01:00
Jonas Bark
eb07b78cce show keys instead of toast 2025-12-07 11:27:14 +01:00
Jonas Bark
f9a13a90a8 fix pairing permission issues 2025-12-07 11:03:21 +01:00
Jonas Bark
6302091c54 improve handling of analog gamepad events 2025-12-07 10:41:21 +01:00
Jonas Bark
1647fe9818 improve handling of analog gamepad events 2025-12-07 10:33:54 +01:00
Jonas Bark
3ece6cb4db cleanup logs, improve handling of analog gamepad events 2025-12-07 10:28:39 +01:00
Jonas Bark
b04732cc24 adjust changelog 2025-12-06 15:43:31 +01:00
Jonas Bark
828819907b refactoring 2025-12-06 15:06:12 +01:00
Jonas Bark
c8c449d2ef screenshot work 2025-12-06 13:42:18 +01:00
Jonas Bark
6b5c202e93 resolve issue #207 2025-12-06 11:56:08 +01:00
Jonas Bark
56c67ae9a5 refactoring 2025-12-06 11:46:18 +01:00
Jonas Bark
94d9467bc3 ui adjustments, fixes 2025-12-06 10:02:56 +01:00
Jonas Bark
fb65616cee create duplicate keymap if needed 2025-12-05 21:35:40 +01:00
Jonas Bark
7e488b3cb1 fix hid device functionality when accessibility service is active 2025-12-05 20:07:48 +01:00
Jonas Bark
9c6084397e fix hid device functionality when accessibility service is active 2025-12-05 20:00:07 +01:00
Jonas Bark
73a0fe6203 fix hid device functionality when accessibility service is active 2025-12-05 19:48:34 +01:00
Jonas Bark
cac1872459 resolve issue #176 2025-12-05 18:08:54 +01:00
Jonas Bark
65f4ca6356 media key detection Windows 2025-12-05 17:57:11 +01:00
Jonas Bark
d771b3da57 cleanup, refactoring 2025-12-05 15:40:04 +01:00
Jonas Bark
d52a062e0c cleanup, refactoring 2025-12-05 13:22:52 +01:00
Jonas Bark
ef350398e1 work on mDNS 2025-12-05 11:14:16 +01:00
Jonas Bark
69fa8834ee work on mDNS 2025-12-05 10:51:40 +01:00
Jonas Bark
ac9bbd3986 work on screenshots 2025-12-04 21:51:16 +00:00
Jonas Bark
6f99f0762a keydown / key up handling for direct connections 2025-12-04 21:17:43 +00:00
Jonas Bark
999fc3faba ui, fixes 2025-12-04 20:59:07 +00:00
Jonas Bark
ebd33666fc ui, fixes 2025-12-04 20:27:53 +00:00
Jonas Bark
8537c5b8c3 ui, fixes 2025-12-04 20:00:01 +00:00
Jonas Bark
3ff59eff6e ui, fixes 2025-12-04 19:42:19 +00:00
Jonas Bark
46752de0e8 ui, fixes 2025-12-04 19:32:02 +00:00
Jonas Bark
27629db924 connection method badges 2025-12-04 14:00:39 +00:00
Jonas Bark
1f00fd9d6c add openbikecontrol as app 2025-12-04 13:27:40 +00:00
Jonas Bark
637bf87dad ui adjustments 2025-12-04 12:58:31 +00:00
Jonas Bark
41415432db ui adjustments 2025-12-01 22:58:29 +00:00
Jonas Bark
745d26dade ui adjustments 2025-12-01 22:01:41 +00:00
Jonas Bark
134de0fcd9 Merge remote-tracking branch 'origin/copilot/ensure-ble-hid-support-windows' into feature/openbikecontrol
# Conflicts:
#	.gitignore
2025-12-01 18:23:14 +00:00
Jonas Bark
ee0c4c083f screenshot adjustments 2025-12-01 18:14:16 +00:00
Jonas Bark
5873ce34e4 translations 2025-12-01 16:58:08 +00:00
Jonas Bark
9904999dd2 translations 2025-12-01 15:46:02 +00:00
Jonas Bark
f5a35ad04b Merge remote-tracking branch 'origin/copilot/extract-translations-to-arb' into feature/openbikecontrol
# Conflicts:
#	lib/pages/trainer.dart
2025-12-01 15:35:54 +00:00
Jonas Bark
7301734b82 zwift keep alive 2025-12-01 15:34:50 +00:00
copilot-swe-agent[bot]
19ada28cf3 Add i18n_extension.dart and fix import/export keys, update imports
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-01 14:48:45 +00:00
copilot-swe-agent[bot]
7655b35b77 Update remaining files to use localized strings
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-01 14:41:28 +00:00
copilot-swe-agent[bot]
0c200eae20 Update widgets and utils to use localized strings
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-01 14:35:11 +00:00
copilot-swe-agent[bot]
18d6c9ec1f Extract translations to intl_en.arb and update pages to use i18n
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-01 14:18:47 +00:00
copilot-swe-agent[bot]
cc16a2c8de Add volume up/down hotkey support for Windows media key detection
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-12-01 14:16:21 +00:00
copilot-swe-agent[bot]
91a6502310 Initial plan 2025-12-01 14:04:35 +00:00
Jonas Bark
bb8720247e add intl 2025-12-01 13:52:20 +00:00
Jonas Bark
b01470339b add intl 2025-12-01 13:50:15 +00:00
Jonas Bark
855fe00346 wording 2025-12-01 13:40:48 +00:00
Jonas Bark
f957258a2c wording 2025-12-01 13:40:00 +00:00
Jonas Bark
945a247274 toast changes 2025-12-01 12:02:41 +00:00
Jonas Bark
59ecfe5186 UI changes 2025-12-01 11:42:54 +00:00
Jonas Bark
dc5e9939c7 UI changes, cleanup 2025-12-01 10:57:23 +00:00
Jonas Bark
79821783e2 auto start connection methods and scanning 2025-12-01 09:30:21 +00:00
jonasbark
531e53eaec Update WINDOWS_STORE_VERSION.txt 2025-11-30 21:08:53 +01:00
Jonas Bark
4f01f23458 Merge branch 'main' into feature/openbikecontrol
# Conflicts:
#	lib/utils/actions/base_actions.dart
2025-11-30 18:46:15 +00:00
Jonas Bark
3760b84fb7 only use direct connection method if they are connected 2025-11-30 18:45:20 +00:00
Jonas Bark
afdb442597 Merge branch 'main' into feature/openbikecontrol
# Conflicts:
#	lib/pages/device.dart
#	lib/widgets/logviewer.dart
2025-11-30 01:09:03 +00:00
Jonas Bark
eadcd17bbb less noise 2025-11-30 01:08:25 +00:00
Jonas Bark
67ca63b047 Merge branch 'feature/kickrbikepro' into feature/openbikecontrol 2025-11-30 01:07:51 +00:00
Jonas Bark
17d3450bd2 fix 2025-11-30 01:07:42 +00:00
Jonas Bark
d7c700a4e5 heureka 2025-11-30 00:44:45 +00:00
Jonas Bark
fd659e40fe heureka 2025-11-29 23:55:43 +00:00
Jonas Bark
bdd7c75ebd attempt to support Wahoo Kickr Bike Pro #195 2025-11-29 21:54:24 +00:00
Jonas Bark
ca5648fec4 attempt to support Wahoo Kickr Bike Pro #195 2025-11-29 19:40:32 +00:00
Jonas Bark
978325ceff attempt to support Wahoo Kickr Bike Pro #195 2025-11-29 19:36:35 +00:00
Jonas Bark
87ad9c630d try kickr 2025-11-29 19:05:38 +00:00
Jonas Bark
6b0ca1b6dd fix 2025-11-29 19:05:22 +00:00
Jonas Bark
3c9a505f4f Merge branch 'main' of github.com:jonasbark/swiftcontrol 2025-11-29 18:42:37 +00:00
Jonas Bark
7bb4ace875 openbikeprotocol Bluetooth emulator 2025-11-29 17:46:12 +00:00
Jonas Bark
499cd86955 openbikeprotocol Bluetooth emulator 2025-11-29 17:10:23 +00:00
Jonas Bark
79bad50ce3 openbikeprotocol emulator 2025-11-29 15:35:09 +00:00
Jonas Bark
2dff9c92fd ftms emulator 2025-11-29 14:45:44 +00:00
Jonas Bark
d05557819e ui adjustments, fixes 2025-11-29 14:37:43 +00:00
Jonas Bark
22a39406ff fix wrong check 2025-11-29 13:51:36 +00:00
Jonas Bark
91d0fd656e consolidate messages 2025-11-29 13:51:05 +00:00
Jonas Bark
4440a04ab8 consolidate messages 2025-11-29 13:46:57 +00:00
Jonas Bark
5bf617d1db move service running message 2025-11-29 13:35:57 +00:00
Jonas Bark
3cf209edd9 less god objects 2025-11-29 13:12:09 +00:00
Jonas Bark
ad95ce23ec add implementation for open bike protocol devices 2025-11-29 12:53:04 +00:00
Jonas Bark
9e29d5dd04 ui adjustments 2025-11-29 10:57:14 +00:00
Jonas Bark
4678ac0255 ui adjustments 2025-11-29 10:54:04 +00:00
Jonas Bark
8660f979b9 ignore device for session fix, logs page 2025-11-29 09:35:38 +00:00
Jonas Bark
c20265ddf8 new ui, BLE fixes 2025-11-28 22:31:18 +00:00
Jonas Bark
2b90606d9c inline permission requests, make them optional unless required 2025-11-28 22:00:48 +00:00
Jonas Bark
c75fd8ab80 Merge remote-tracking branch 'origin/main' into feature/openbikecontrol
# Conflicts:
#	lib/widgets/menu.dart
2025-11-28 20:21:32 +00:00
Jonas Bark
fc13d71960 version++ 2025-11-28 18:50:56 +00:00
Jonas Bark
354179de30 potential fixes for #196 2025-11-28 18:49:55 +00:00
Jonas Bark
e4ea924b1a fix Windows app icon 2025-11-28 17:49:49 +00:00
Jonas Bark
f9b5dda123 new UI 2025-11-28 08:31:34 +00:00
Jonas Bark
4cd38d4502 new UI 2025-11-28 00:26:21 +00:00
Jonas Bark
70c9a5b6a3 new UI 2025-11-27 23:38:00 +00:00
Jonas Bark
4fb46de074 new UI 2025-11-27 23:31:29 +00:00
Jonas Bark
b0487f534c new UI 2025-11-27 22:36:19 +00:00
Jonas Bark
719eb9b50f new UI 2025-11-27 19:59:49 +00:00
Jonas Bark
f3148405d7 Merge branch 'main' into feature/openbikecontrol 2025-11-27 18:10:26 +00:00
Jonas Bark
6da13f18af UI fix 2025-11-27 17:03:14 +00:00
Jonas Bark
b3b6f510d8 work on new ui 2025-11-27 13:22:14 +00:00
Jonas Bark
01400ab955 work on new ui 2025-11-27 12:49:21 +00:00
Jonas Bark
8348848ef8 work on new ui 2025-11-27 09:44:26 +00:00
Jonas Bark
04920e7eee Merge remote-tracking branch 'origin/main' 2025-11-26 22:34:20 +00:00
Jonas Bark
464c5208dc error logging 2025-11-26 22:34:11 +00:00
jonasbark
981254875b Update README.md 2025-11-26 10:32:29 +01:00
Jonas Bark
d941135fff remove media key detection due to Apple restrictions 2025-11-25 19:26:39 +00:00
jonasbark
28663f5cdf Update Windows Store version to 3.6.0 2025-11-24 18:14:41 +01:00
Jonas Bark
bfc12711ad screenshot adjustments 2025-11-23 22:03:21 +00:00
Jonas Bark
076e729e39 screenshot adjustments 2025-11-23 21:56:21 +00:00
Jonas Bark
570f5ca82d missing BikeControl renaming 2025-11-23 21:33:56 +00:00
Jonas Bark
9749a9018d missing BikeControl renaming 2025-11-23 21:26:55 +00:00
Jonas Bark
4724f55e6b some UI adjustments 2025-11-23 21:24:18 +00:00
Jonas Bark
8f68636bfc some UI changes 2025-11-23 21:06:37 +00:00
Jonas Bark
fefcb1db53 status widget for accessibility service 2025-11-23 20:57:52 +00:00
Jonas Bark
98fd5c5d7c status widget 2025-11-23 20:28:28 +00:00
Jonas Bark
e97a76e488 better error handling 2025-11-23 08:56:30 +00:00
Jonas Bark
02f25899e9 show action results in the UI 2025-11-23 08:33:19 +00:00
Jonas Bark
3a0ef5110d more name change work 2025-11-23 08:08:03 +00:00
Jonas Bark
36c5df7d03 Merge remote-tracking branch 'origin/copilot/rename-app-to-bikecontrol' 2025-11-22 18:10:36 +00:00
copilot-swe-agent[bot]
4252ed05ae Complete remaining SwiftControl to BikeControl renames in code
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-21 18:11:41 +00:00
copilot-swe-agent[bot]
2adf3026fd Rename app from SwiftControl to BikeControl and update GitHub repo references
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-21 18:07:48 +00:00
copilot-swe-agent[bot]
2cc705c556 Initial plan 2025-11-21 17:57:39 +00:00
Jonas Bark
faff8ab219 adjust changelog 2025-11-21 07:09:00 +00:00
Jonas Bark
e63e647ebc better formatting 2025-11-20 23:09:56 +00:00
Jonas Bark
e2339321bb Merge branch 'main' of github.com:jonasbark/swiftcontrol 2025-11-20 23:04:57 +00:00
jonasbark
f7c57b1f15 Merge pull request #192 from jonasbark/copilot/add-predefined-actions-keymaps
Add predefined action selector for custom keymaps from selected trainer app
2025-11-20 23:04:28 +00:00
copilot-swe-agent[bot]
a1fad509cf Optimize to avoid redundant settings.getTrainerApp() calls
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-20 23:00:03 +00:00
copilot-swe-agent[bot]
daec33f827 Handle empty actions list and simplify key label logic
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-20 22:58:10 +00:00
copilot-swe-agent[bot]
cf66845a38 Address code review feedback - improve list copying and clarify key formatting
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-20 22:56:27 +00:00
copilot-swe-agent[bot]
7199f5b545 Use current trainer app's keymap instead of showing dialog
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-20 22:54:54 +00:00
copilot-swe-agent[bot]
0490956551 Apply performance optimizations with const widgets
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-20 22:53:18 +00:00
copilot-swe-agent[bot]
8be2108cdc Address code review feedback - improve type checking and readability
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-20 22:52:02 +00:00
copilot-swe-agent[bot]
4595e509c9 Add predefined action selector for custom keymaps
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-20 22:49:49 +00:00
copilot-swe-agent[bot]
17d7953d11 Initial plan 2025-11-20 22:44:31 +00:00
Jonas Bark
4adfe0812d fix compatibility checks on this device on iOS devices 2025-11-20 21:34:03 +00:00
Jonas Bark
7ca9c8752b Merge remote-tracking branch 'origin/main' 2025-11-18 22:45:56 +00:00
Jonas Bark
ab53d23404 add keyboard controls to Rouvys virtual shifting 2025-11-18 22:45:47 +00:00
jonasbark
9894271145 Update Windows Store version to 3.5.0 2025-11-18 23:11:14 +01:00
Jonas Bark
5d9960156c update rouvy keymap to support their latest version 2025-11-18 22:08:16 +00:00
Jonas Bark
aa6782d29b Merge branch 'copilot/add-screenshots-to-release' 2025-11-18 21:54:08 +00:00
Jonas Bark
98d683a6a5 Merge remote-tracking branch 'origin/main' 2025-11-18 21:54:00 +00:00
jonasbark
55a2e4db79 Clarify device troubleshooting headings
Updated headings to include 'Ride' alongside 'Click' for clarity.
2025-11-17 21:54:00 +01:00
Jonas Bark
4db985e2e5 screenshot adjustments 2025-11-17 14:33:23 +01:00
Jonas Bark
bbd95beb36 add bluetooth device detection unit test 2025-11-17 14:18:34 +01:00
Jonas Bark
a9ee0dc9a1 fix unit tests 2025-11-17 14:01:29 +01:00
Jonas Bark
46d3770a28 more work on screenshots 2025-11-17 13:55:27 +01:00
Jonas Bark
99ee63ce1f more work on screenshots 2025-11-17 13:14:07 +01:00
Jonas Bark
cb10ad685e Merge branch 'main' into copilot/add-screenshots-to-release 2025-11-17 12:49:18 +01:00
Jonas Bark
748a21fb54 small refactoring 2025-11-17 12:49:09 +01:00
Jonas Bark
b3ffe867c6 version++ 2025-11-16 21:32:49 +01:00
Jonas Bark
fb1ffec37d version++ 2025-11-16 21:04:28 +01:00
Jonas Bark
9ea4f7157a some possible fixes 2025-11-16 20:53:56 +01:00
Jonas Bark
a9b43bd347 windows build fix 2025-11-16 12:51:32 +01:00
Jonas Bark
b9ac193e77 windows build fix 2025-11-16 12:47:13 +01:00
Jonas Bark
1d4947b3ae windows build fix 2025-11-16 12:46:17 +01:00
Jonas Bark
74e098e9b1 adjust integration tests 2025-11-16 12:42:33 +01:00
Jonas Bark
e5dae225f1 Merge branch 'main' into copilot/add-screenshots-to-release
# Conflicts:
#	lib/bluetooth/connection.dart
#	lib/utils/settings/settings.dart
2025-11-16 12:30:21 +01:00
Jonas Bark
0339089972 version++ 2025-11-16 11:58:05 +01:00
Jonas Bark
1e8bd61264 update changelog 2025-11-16 10:20:47 +01:00
Jonas Bark
79613bc8de resolve issue #179 2025-11-16 10:15:46 +01:00
Jonas Bark
d0ec785e32 remove settings file when corrupted #180 2025-11-16 10:13:37 +01:00
Jonas Bark
020b91fd21 Merge branch 'main' of github.com:jonasbark/swiftcontrol 2025-11-15 12:54:53 +01:00
Jonas Bark
f2406152fd Merge branch 'copilot/update-cycplus-bc2-implementation' 2025-11-15 12:54:45 +01:00
Jonas Bark
ab3ef7be53 resolve #186 2025-11-15 12:54:30 +01:00
jonasbark
bb7484ff2e Merge pull request #185 from jonasbark/copilot/update-cycplus-bc2-implementation
Simplify Cycplus BC2 implementation to match reference state machine
2025-11-15 10:35:28 +01:00
copilot-swe-agent[bot]
80061fd076 Update Cycplus BC2 implementation to match reference
- Only look at bytes at index 6 and 7 (no full frame parsing)
- Implement state machine for pressed/released states
- Track state independently for each index
- Trigger on state transitions (pressed to different pressed)
- Reset state on release (0x00) or after successful trigger

Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-15 09:27:45 +00:00
Jonas Bark
124e005fb1 Merge branch 'main' of github.com:jonasbark/swiftcontrol 2025-11-15 10:27:11 +01:00
Jonas Bark
8e760ef202 more error checking 2025-11-15 10:27:07 +01:00
copilot-swe-agent[bot]
de740f6453 Initial plan 2025-11-15 09:20:32 +00:00
jonasbark
3bde90ae62 Merge pull request #182 from jonasbark/copilot/fix-ble-device-reconnection
Persist ignored BLE devices across app restarts
2025-11-15 09:04:53 +01:00
copilot-swe-agent[bot]
aee8dc2e07 Make getIgnoredDeviceIds and getIgnoredDeviceNames private
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-15 07:57:35 +00:00
jonasbark
b3952542f8 Fix grammar and formatting in README.md 2025-11-14 15:08:05 +01:00
Jonas Bark
910f23a3f6 Merge branch 'copilot/fix-cycplus-bc2-button-logic' 2025-11-14 14:57:47 +01:00
Jonas Bark
5cc9ac85af attempt to handle https://github.com/jonasbark/swiftcontrol/issues/179 2025-11-14 14:57:35 +01:00
copilot-swe-agent[bot]
e10e22d038 Add support for 0xfe button codes in CYCPLUS BC2 device
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-14 13:34:36 +00:00
copilot-swe-agent[bot]
aaeaec36a2 Initial plan 2025-11-14 13:29:59 +00:00
Jonas Bark
c16a593f3c potential fix for https://github.com/jonasbark/swiftcontrol/issues/183 2025-11-14 14:28:47 +01:00
Jonas Bark
50d9f47576 potential fix for https://github.com/jonasbark/swiftcontrol/issues/183 2025-11-14 12:23:30 +01:00
copilot-swe-agent[bot]
a53fb578ef Fix disconnect logic and improve UI flow
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-13 15:25:14 +00:00
copilot-swe-agent[bot]
1f89859a03 Implement persistent ignored devices feature
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-13 15:23:12 +00:00
copilot-swe-agent[bot]
7c1cee6748 Initial plan 2025-11-13 15:17:47 +00:00
Jonas Bark
a4949ad615 error handling 2025-11-12 22:22:44 +01:00
Jonas Bark
a57f4654b0 possible fix for https://github.com/jonasbark/swiftcontrol/issues/180#issuecomment-3523836191 2025-11-12 21:49:01 +01:00
Jonas Bark
8192d3addf show to user when MyWhoosh Link can't be started, cleanup 2025-11-12 09:22:41 +01:00
Jonas Bark
308f461ad4 show to user when MyWhoosh Link can't be started, cleanup 2025-11-12 09:22:19 +01:00
Jonas Bark
513b2ba367 screenshot creation #1 2025-11-11 10:25:56 +01:00
Jonas Bark
69f47fa984 bluetooth naming 2025-11-11 09:06:07 +01:00
copilot-swe-agent[bot]
4daf553514 Add documentation for screenshot generation process
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-11 07:28:00 +00:00
copilot-swe-agent[bot]
aeae148e0b Implement dynamic screenshot generation during build pipeline
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-11 07:26:47 +00:00
copilot-swe-agent[bot]
1a9a265671 Add screenshot packaging and release attachment to build workflow
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-11 07:19:13 +00:00
copilot-swe-agent[bot]
b06e9ad4b3 Initial plan 2025-11-11 07:12:19 +00:00
copilot-swe-agent[bot]
7ecce43a3d Add documentation for Windows global media key detection
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-10 07:53:16 +00:00
copilot-swe-agent[bot]
82cb41c207 Remove CodeQL artifact from repository
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-10 07:52:26 +00:00
copilot-swe-agent[bot]
c356242a8f Improve hotkey registration error handling
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-10 07:52:01 +00:00
copilot-swe-agent[bot]
61ad7c2eef Implement global media key detection for Windows using RegisterHotKey API
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-10 07:49:27 +00:00
copilot-swe-agent[bot]
311104ddf0 Initial plan 2025-11-10 07:44:08 +00:00
Jonas Bark
2b106fd1c9 change mail address 2025-11-09 19:33:32 +01:00
Jonas Bark
1784e008ee a few fixes 2025-11-09 19:17:35 +01:00
Jonas Bark
33ccdbd7af a few fixes 2025-11-09 16:38:13 +01:00
Jonas Bark
d15f1ddc13 Merge branch 'main' of github.com:jonasbark/swiftcontrol 2025-11-09 16:14:34 +01:00
Jonas Bark
e7ea01cd60 Merge branch 'copilot/create-landing-page-for-apps' 2025-11-09 16:14:29 +01:00
Jonas Bark
f7ed426441 cleanup 2025-11-09 16:14:15 +01:00
Jonas Bark
d30485b82e add a clarifying selection to help users choose where SwiftControl should run 2025-11-09 16:13:07 +01:00
Jonas Bark
4e646ab922 add dark mode, logs to the support entry, link to new landing page 2025-11-09 15:57:54 +01:00
jonasbark
4183ede58d Updated .gitignore 2025-11-09 08:54:17 +01:00
copilot-swe-agent[bot]
dd85e99e4b Update landing page per feedback: blue background, black headings, restructured compatibility checker, and instructions section
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-09 07:42:42 +00:00
jonasbark
2334a88452 Merge pull request #174 from jonasbark/copilot/add-changelog-to-releases
Add changelog to GitHub release body
2025-11-09 08:23:40 +01:00
copilot-swe-agent[bot]
ee64b18f75 Refactor: reuse get_latest_changelog.sh output
- Removed duplicate awk logic from generate_release_body.sh
- Now calls get_latest_changelog.sh for changelog content
- Simplified script by removing unused variables
- Adds version header separately for GitHub releases

Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-09 07:21:17 +00:00
copilot-swe-agent[bot]
647dac9e7c Add interactive landing page with compatibility checker
Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-09 07:14:46 +00:00
copilot-swe-agent[bot]
df2496eb67 Add changelog to GitHub release body
- Created generate_release_body.sh script to combine changelog with store links
- Updated build.yml workflow to use new release body
- Updated patch.yml workflow to use new release body
- Release body now includes latest changelog entry followed by store download links

Co-authored-by: jonasbark <1151304+jonasbark@users.noreply.github.com>
2025-11-09 07:12:43 +00:00
copilot-swe-agent[bot]
859424b895 Initial plan 2025-11-09 07:08:36 +00:00
copilot-swe-agent[bot]
dde3f38bde Initial plan 2025-11-09 07:07:37 +00:00
261 changed files with 17221 additions and 5481 deletions

View File

@@ -36,7 +36,7 @@ on:
env:
SHOREBIRD_TOKEN: ${{ secrets.SHOREBIRD_TOKEN }}
FLUTTER_VERSION: 3.35.5
FLUTTER_VERSION: 3.38.4
jobs:
build:
@@ -102,12 +102,24 @@ jobs:
with:
cache: true
- name: Set Up Flutter
uses: subosito/flutter-action@v2
with:
channel: 'stable'
flutter-version: ${{ env.FLUTTER_VERSION }}
- name: Generate translation files
run: |
flutter pub global activate intl_utils;
flutter pub global run intl_utils:generate;
- name: 🚀 Shorebird Release macOS
if: inputs.build_mac
uses: shorebirdtech/shorebird-release@v1
with:
flutter-version: ${{ env.FLUTTER_VERSION }}
platform: macos
args: "--dart-define=VERIFYING_SHARED_SECRET=${{ secrets.VERIFYING_SHARED_SECRET }}"
- name: Decode Keystore
if: inputs.build_android
@@ -123,13 +135,6 @@ jobs:
platform: android
args: "--artifact=apk"
- name: Set Up Flutter
if: inputs.build_web
uses: subosito/flutter-action@v2
with:
channel: 'stable'
flutter-version: ${{ env.FLUTTER_VERSION }}
- name: Build Web
if: inputs.build_web
run: flutter build web --release --base-href "/swiftcontrol/"
@@ -152,13 +157,19 @@ jobs:
mkdir -p whatsnew
./scripts/get_latest_changelog.sh | head -c 500 > whatsnew/whatsnew-en-US
- name: Generate release body
if: inputs.build_github
run: |
chmod +x scripts/generate_release_body.sh
./scripts/generate_release_body.sh > /tmp/release_body.md
- name: 🚀 Shorebird Release iOS
if: inputs.build_ios
uses: shorebirdtech/shorebird-release@v1
with:
flutter-version: ${{ env.FLUTTER_VERSION }}
platform: ios
args: "--export-options-plist ios/ExportOptions.plist"
args: "--export-options-plist ios/ExportOptions.plist --dart-define=VERIFYING_SHARED_SECRET=${{ secrets.VERIFYING_SHARED_SECRET }}"
- name: Prepare App Store authentication key
if: inputs.build_ios || inputs.build_mac
@@ -176,7 +187,7 @@ jobs:
serviceAccountJsonPlainText: ${{ secrets.SERVICE_ACCOUNT_JSON }}
packageName: de.jonasbark.swiftcontrol
releaseFiles: build/app/outputs/bundle/release/app-release.aab
track: production
track: ${{ github.ref == 'refs/heads/main' && 'production' || 'alpha' }}
whatsNewDirectory: whatsnew
- name: Upload to macOS App Store
@@ -185,8 +196,8 @@ jobs:
APPSTORE_API_KEY: ${{ secrets.APPSTORE_API_KEY }}
APPSTORE_API_ISSUER_ID: ${{ secrets.APPSTORE_API_ISSUER_ID }}
run: |
productbuild --component "build/macos/Build/Products/Release/SwiftControl.app" /Applications "SwiftControl.pkg" --sign "3rd Party Mac Developer Installer: JONAS TASSILO BARK (UZRHKPVWN9)";
xcrun altool --upload-app -f SwiftControl.pkg -t osx --apiKey "$APPSTORE_API_KEY" --apiIssuer "$APPSTORE_API_ISSUER_ID";
productbuild --component "build/macos/Build/Products/Release/BikeControl.app" /Applications "BikeControl.pkg" --sign "3rd Party Mac Developer Installer: JONAS TASSILO BARK (UZRHKPVWN9)";
xcrun altool --upload-app -f BikeControl.pkg -t osx --apiKey "$APPSTORE_API_KEY" --apiIssuer "$APPSTORE_API_ISSUER_ID";
- name: Upload to iOS App Store
if: inputs.build_ios
@@ -197,40 +208,40 @@ jobs:
xcrun altool --upload-app -f build/ios/ipa/swift_play.ipa -t ios --apiKey "$APPSTORE_API_KEY" --apiIssuer "$APPSTORE_API_ISSUER_ID";
- name: Handle Android archives
if: inputs.build_android && inputs.build_github
if: inputs.build_android && inputs.build_github && false
run: |
cp build/app/outputs/flutter-apk/app-release.apk build/app/outputs/flutter-apk/SwiftControl.android.apk
cp build/app/outputs/flutter-apk/app-release.apk build/app/outputs/flutter-apk/BikeControl.android.apk
- name: Code Signing of macOS app
if: inputs.build_mac && inputs.build_github
run: /usr/bin/codesign --deep --force -s "$DEVELOPER_ID_APPLICATION_SIGNING_IDENTITY" --entitlements ../../../../../macos/Runner/Release.entitlements --options runtime SwiftControl.app -v
if: inputs.build_mac && inputs.build_github && false
run: /usr/bin/codesign --deep --force -s "$DEVELOPER_ID_APPLICATION_SIGNING_IDENTITY" --entitlements ../../../../../macos/Runner/Release.entitlements --options runtime BikeControl.app -v
working-directory: build/macos/Build/Products/Release
env:
DEVELOPER_ID_APPLICATION_SIGNING_IDENTITY: ${{ secrets.DEVELOPER_ID_APPLICATION_SIGNING_IDENTITY }}
- name: Handle macOS archives
if: inputs.build_mac && inputs.build_github
if: inputs.build_mac && inputs.build_github && false
run: |
cd build/macos/Build/Products/Release/
zip -r SwiftControl.macos.zip SwiftControl.app/
zip -r BikeControl.macos.zip BikeControl.app/
- name: Upload Android Artifacts
if: inputs.build_android && inputs.build_github
if: inputs.build_android && inputs.build_github && false
uses: actions/upload-artifact@v4
with:
overwrite: true
name: Releases
path: |
build/app/outputs/flutter-apk/SwiftControl.android.apk
build/app/outputs/flutter-apk/BikeControl.android.apk
- name: Upload macOS Artifacts
if: inputs.build_mac && inputs.build_github
if: inputs.build_mac && inputs.build_github && false
uses: actions/upload-artifact@v4
with:
overwrite: true
name: Releases
path: |
build/macos/Build/Products/Release/SwiftControl.macos.zip
build/macos/Build/Products/Release/BikeControl.macos.zip
#10 Extract Version
- name: Extract version from pubspec.yaml
@@ -245,29 +256,47 @@ jobs:
if: inputs.build_github
uses: ncipollo/release-action@v1
with:
artifacts: "build/app/outputs/flutter-apk/SwiftControl.android.apk,build/macos/Build/Products/Release/SwiftControl.macos.zip"
artifacts: "build/app/outputs/flutter-apk/BikeControl.android.apk,build/macos/Build/Products/Release/BikeControl.macos.zip"
allowUpdates: true
prerelease: true
bodyFile: scripts/RELEASE_NOTES.md
bodyFile: /tmp/release_body.md
tag: v${{ env.VERSION }}
token: ${{ secrets.TOKEN }}
windows:
needs: build
if: inputs.build_windows
name: Build & Release on Windows
runs-on: windows-latest
runs-on: windows-2025
steps:
#1 Checkout Repository
- name: Checkout Repository
uses: actions/checkout@v3
- name: Extract version from pubspec.yaml (Windows)
shell: pwsh
run: |
$version = Select-String '^version: ' pubspec.yaml | ForEach-Object {
($_ -split ' ')[1].Trim()
}
echo "VERSION=$version" >> $env:GITHUB_ENV
- name: 🐦 Setup Shorebird
uses: shorebirdtech/setup-shorebird@v1
with:
cache: true
- name: Set Up Flutter
uses: subosito/flutter-action@v2
with:
channel: 'stable'
flutter-version: ${{ env.FLUTTER_VERSION }}
- name: Generate translation files
run: |
flutter pub global activate intl_utils;
flutter pub global run intl_utils:generate;
- name: 🚀 Shorebird Release Windows
uses: shorebirdtech/shorebird-release@v1
with:
@@ -296,7 +325,7 @@ jobs:
Write-Warning "$dll not found in $source"
}
}
Compress-Archive -Path "build/windows/x64/runner/Release/*" -DestinationPath "build/windows/x64/runner/Release/SwiftControl.windows.zip"
Compress-Archive -Path "build/windows/x64/runner/Release/*" -DestinationPath "build/windows/x64/runner/Release/bike_control.windows.zip"
- uses: microsoft/setup-msstore-cli@v1
if: false
@@ -305,12 +334,6 @@ jobs:
if: false
run: msstore reconfigure --tenantId $ --clientId $ --clientSecret $ --sellerId $
- name: Set Up Flutter
uses: subosito/flutter-action@v2
with:
channel: 'stable'
flutter-version: ${{ env.FLUTTER_VERSION }}
- name: Create MSIX package
run: dart run msix:create
@@ -318,35 +341,20 @@ jobs:
if: false
run: msstore publish -v "build/windows/x64/runner/Release/"
- name: Rename swift_control.msix to SwiftControl.windows.msix
shell: pwsh
run: |
Rename-Item -Path "build/windows/x64/runner/Release/swift_control.msix" -NewName "SwiftControl.windows.msix"
- name: Upload Artifacts
uses: actions/upload-artifact@v4
with:
overwrite: true
name: Releases
path: |
build/windows/x64/runner/Release/SwiftControl.windows.zip
build/windows/x64/runner/Release/SwiftControl.windows.msix
- name: Extract version from pubspec.yaml (Windows)
shell: pwsh
run: |
$version = Select-String '^version: ' pubspec.yaml | ForEach-Object {
($_ -split ' ')[1].Trim()
}
echo "VERSION=$version" >> $env:GITHUB_ENV
build/windows/x64/runner/Release/bike_control.windows.zip
build/windows/x64/runner/Release/bike_control.msix
- name: Update Release
uses: ncipollo/release-action@v1
with:
allowUpdates: true
artifacts: "build/windows/x64/runner/Release/SwiftControl.windows.zip,build/windows/x64/runner/Release/SwiftControl.windows.msix"
bodyFile: scripts/RELEASE_NOTES.md
artifacts: "build/windows/x64/runner/Release/bike_control.msix"
prerelease: true
tag: v${{ env.VERSION }}
token: ${{ secrets.TOKEN }}

View File

@@ -5,7 +5,7 @@ on:
env:
SHOREBIRD_TOKEN: ${{ secrets.SHOREBIRD_TOKEN }}
FLUTTER_VERSION: 3.35.5
FLUTTER_VERSION: 3.38.5
jobs:
build:
@@ -27,6 +27,17 @@ jobs:
with:
cache: true
- name: Set Up Flutter
uses: subosito/flutter-action@v2
with:
channel: 'stable'
flutter-version: ${{ env.FLUTTER_VERSION }}
- name: Generate translation files
run: |
flutter pub global activate intl_utils;
flutter pub global run intl_utils:generate;
- name: Install certificates
env:
DEVELOPER_ID_APPLICATION_P12_BASE64_MAC: ${{ secrets.DEVELOPER_ID_APPLICATION_P12_BASE64_MAC }}
@@ -96,42 +107,46 @@ jobs:
release-version: latest
args: '--allow-asset-diffs --allow-native-diffs'
- name: Set Up Flutter
uses: subosito/flutter-action@v2
with:
channel: 'stable'
flutter-version: ${{ env.FLUTTER_VERSION }}
# shorebird struggles with the app from GitHub
- name: Build macOS
if: false
run: flutter build macos --release;
- name: Sign macOS build
if: false
env:
DEVELOPER_ID_APPLICATION_SIGNING_IDENTITY: ${{ secrets.DEVELOPER_ID_APPLICATION_SIGNING_IDENTITY }}
run: |
version=$(grep '^version: ' pubspec.yaml | cut -d ' ' -f 2 | tr -d '\r');
echo "VERSION=$version" >> $GITHUB_ENV;
cd build/macos/Build/Products/Release/;
/usr/bin/codesign --deep --force -s "$DEVELOPER_ID_APPLICATION_SIGNING_IDENTITY" --entitlements ../../../../../macos/Runner/Release.entitlements --options runtime SwiftControl.app -v;
zip -r SwiftControl.macos.zip SwiftControl.app/;
/usr/bin/codesign --deep --force -s "$DEVELOPER_ID_APPLICATION_SIGNING_IDENTITY" --entitlements ../../../../../macos/Runner/Release.entitlements --options runtime BikeControl.app -v;
zip -r BikeControl.macos.zip BikeControl.app/;
#9 Upload Artifacts
- name: Upload Artifacts
if: false
uses: actions/upload-artifact@v4
with:
overwrite: true
name: Releases
path: |
build/macos/Build/Products/Release/SwiftControl.macos.zip
build/macos/Build/Products/Release/BikeControl.macos.zip
- name: Generate release body
if: false
run: |
chmod +x scripts/generate_release_body.sh
./scripts/generate_release_body.sh > /tmp/release_body.md
# add artifact to release
- name: Create Release
if: false
uses: ncipollo/release-action@v1
with:
allowUpdates: true
artifacts: "build/macos/Build/Products/Release/SwiftControl.macos.zip"
bodyFile: scripts/RELEASE_NOTES.md
artifacts: "build/macos/Build/Products/Release/BikeControl.macos.zip"
bodyFile: /tmp/release_body.md
prerelease: true
tag: v${{ env.VERSION }}
token: ${{ secrets.TOKEN }}
@@ -145,18 +160,22 @@ jobs:
- name: Checkout Repository
uses: actions/checkout@v3
#2 Setup Java
- name: Set Up Java
uses: actions/setup-java@v3.12.0
with:
distribution: 'oracle'
java-version: '17'
- name: 🐦 Setup Shorebird
uses: shorebirdtech/setup-shorebird@v1
with:
cache: true
- name: Set Up Flutter
uses: subosito/flutter-action@v2
with:
channel: 'stable'
flutter-version: ${{ env.FLUTTER_VERSION }}
- name: Generate translation files
run: |
flutter pub global activate intl_utils;
flutter pub global run intl_utils:generate;
- name: 🚀 Shorebird Patch Windows
uses: shorebirdtech/shorebird-patch@v1
with:

View File

@@ -34,6 +34,12 @@ jobs:
with:
channel: 'stable'
- name: Generate translation files
run: |
flutter pub global activate intl_utils;
flutter pub global run intl_utils:generate;
#4 Install Dependencies
- name: Install Dependencies
run: flutter pub get

5
.gitignore vendored
View File

@@ -41,10 +41,15 @@ app.*.symbols
# Obfuscation related
app.*.map.json
localazy.json
# Android Studio will place build artifacts here
/android/app/debug
/android/app/profile
/android/app/release
lib/gen/
service-account.json
.env
lib/generated

3
.vscode/launch.json vendored
View File

@@ -5,7 +5,8 @@
{
"name": "swiftcontrol",
"request": "launch",
"type": "dart"
"type": "dart",
"program": "lib/main.dart"
},
{
"name": "swiftcontrol (profile mode)",

View File

@@ -1,3 +1,60 @@
### 4.2.0 (20-12-2025)
BikeControl now offers a free trial period of 5 days for all features, so you can test everything before deciding to purchase a license. Please contact the support if you experience any issues!
**Features**:
- support for SRAM AXS/eTap
- only single or double click is supported (no individual button mapping possible, yet)
- use your phone/tablet for steering by attaching your device on your handlebar!
- App is now available in Polish (thanks to Wandrocet)
**Fixes**:
- You will now be notified when a connection to your controller is lost
- improved UI of the Keymap customization screen
### 4.1.0 (16-12-2025)
**Features**:
- control your trainer manually without requiring a controller - just like a Companion app
- support for Wahoo KICKR HEADWIND: control the fan via your controller
**Fixes**:
- Gamepads: handle analog values correctly on Windows
- MyWhoosh: updated default keymap to use the new A+D keys for steering
### 4.0.0 (07-12-2025)
- a brand-new design
- Accessibility Permission is now optional on Android
- Zwift is now fully supported on all operating systems
- you can choose between network based control or bluetooth based control
- MyWhoosh can now also be controlled with BikeControl running on the same iPad / iPhone
- Translations available in German and French
- support for Wahoo KICKR BIKE PRO
- support for the OpenBikeControl protocol for supported Trainer apps
- this enables seamless and official integration, independent of the operating system
- learn more at https://openbikecontrol.org
### 3.6.0 (23-11-2025)
SwiftControl is now called BikeControl!
**Features:**
- show a list of predefined keymaps for the selected trainer app when using a custom keymap
- status icons so it's clear what's missing
**Fixes:**
- Update Rouvy keymap to support virtual shifting in their latest version
### 3.5.0 (16-11-2025)
**New Features:**
- Dark mode support
- Cycplus BC2 support (thanks @schneewoehner)
- Ignored devices now persist across app restarts - remove them from ignored devices via the menu
**Fixes:**
- resolve issues during app start
### 3.4.0 (08-11-2025)
**New Features:**
- Support for Shimano Di2
@@ -7,7 +64,7 @@
**Fixes:**
- fix detection of Elite Square Sterzo devices
- recognize cheap Bluetooth device clicks also when SwiftControl is in the background
- recognize cheap Bluetooth device clicks also when BikeControl is in the background
### 3.3.0 (31-10-2025)
@@ -29,7 +86,7 @@
### 3.2.0 (2025-10-22)
- a brand-new way of controlling MyWhoosh:
- device pairing no longer required as mouse emulation is no longer needed
- SwiftControl can now stay in the background
- BikeControl can now stay in the background
- more devices can be controlled
- do more, such as define Emotes, Camera angles and steering
@@ -39,16 +96,16 @@
- support for Wahook Kickr Bike Shift (thanks @MattW2)
- initial support for Elite Square Smart Frame
- reconnects to your device automatically when connection is lost
- SwiftControl now warns you if your device firmware is outdated
- SwiftControl is now available in Microsoft Store: https://apps.microsoft.com/detail/9NP42GS03Z26
- BikeControl now warns you if your device firmware is outdated
- BikeControl is now available in Microsoft Store: https://apps.microsoft.com/detail/9NP42GS03Z26
### 3.0.3 (2025-10-12)
- SwiftControl now supports iOS!
- Note that you can't run SwiftControl and your trainer app on the same iPhone due to iOS limitations but...:
- You can now use SwiftControl as "remote control" for other devices, such as an iPad. Example scenario:
- your phone (Android/iOS) runs SwiftControl and connects to your Click devices
- your iPad or other tablet runs e.g. MyWhoosh (does not need to have SwiftControl installed)
- after pairing SwiftControl to your iPad / tablet via Bluetooth your phone will send the button presses to your iPad / tablet
- BikeControl now supports iOS!
- Note that you can't run BikeControl and your trainer app on the same iPhone due to iOS limitations but...:
- You can now use BikeControl as "remote control" for other devices, such as an iPad. Example scenario:
- your phone (Android/iOS) runs BikeControl and connects to your Click devices
- your iPad or other tablet runs e.g. MyWhoosh (does not need to have BikeControl installed)
- after pairing BikeControl to your iPad / tablet via Bluetooth your phone will send the button presses to your iPad / tablet
- Ride: analog paddles are now supported thanks to contributor @jmoro
- you can now zoom in and out in the Keymap customization screen
@@ -68,8 +125,8 @@
### 2.5.0 (2025-09-25)
- Improve usability
- SwiftControl is now available via the Play Store: https://play.google.com/store/apps/details?id=de.jonasbark.swiftcontrol
- SwiftControl will continue to be available to download for free on GitHub
- BikeControl is now available via the Play Store: https://play.google.com/store/apps/details?id=de.jonasbark.swiftcontrol
- BikeControl will continue to be available to download for free on GitHub
- contact me if you already donated and I'll get a voucher for you :)
### 2.4.0+1 (2025-09-17)

View File

@@ -1 +0,0 @@
Instructions will be added soon

View File

@@ -1,12 +1 @@
**Instructions for using the MyWhoosh Direct Connect method**
1) launch MyWhoosh on the device of your choice
2) launch MyWhoosh Link, check if the "Link" connection works
3) close MyWhoosh Link
4) open SwiftControl, follow on screen instructions
Once you've confirmed the connection in SwiftControl you won't have to repeat step 2 and 3 again in the future. This is just to make sure the connection works in general.
And here's a video with a few explanations:
[![SwiftControl Instruction for iOS](https://img.youtube.com/vi/p8sgQhuufeI/0.jpg)](https://www.youtube.com/watch?v=p8sgQhuufeI)
[https://www.youtube.com/watch?v=p8sgQhuufeI](https://www.youtube.com/watch?v=p8sgQhuufeI)
Moved to [INSTRUCTIONS_MYWHOOSH_LINK.md](INSTRUCTIONS_MYWHOOSH_LINK.md)

View File

@@ -1 +0,0 @@
Instructions will be added soon

View File

@@ -0,0 +1,31 @@
## Instructions for using the MyWhoosh "Link" connection method
*
1) launch MyWhoosh on the device of your choice
2) launch MyWhoosh Link, check if the "Link" connection works
3) **close MyWhoosh Link** - very important!
4) open BikeControl, follow on screen instructions
Once you've confirmed the connection in BikeControl you won't have to repeat step 2 and 3 again in the future. This is just to make sure the connection works in general.
And here's a video with a few explanations:
[![BikeControl Instruction for iOS](https://img.youtube.com/vi/p8sgQhuufeI/0.jpg)](https://www.youtube.com/watch?v=p8sgQhuufeI)
[https://www.youtube.com/watch?v=p8sgQhuufeI](https://www.youtube.com/watch?v=p8sgQhuufeI)
## The MyWhoosh Link app itself works fine, but BikeControl doesn't connect
*
The MyWhoosh Link app must not run simultaneously with BikeControl. Make sure the MyWhoosh Link app is fully closed, then reopen BikeControl and try connecting again.
## MyWhoosh "Link" method never connects
*
The same network restrictions apply for BikeControl as it applies to MyWhoosh Link app. Please verify with the MyWhoosh Link app if a connection is possible at all.
Here are some instructions that can help:
[https://mywhoosh.com/troubleshoot/](https://mywhoosh.com/troubleshoot/)
[https://www.facebook.com/groups/mywhoosh/posts/1323791068858873/](https://www.facebook.com/groups/mywhoosh/posts/1323791068858873/)
In essence:
- your two devices (phone, tablet) need to be on the same WiFi network
- on iOS you have to turn off "Private Wi-Fi Address" in the WiFi settings
- Limit IP Address Tracking may need to be disabled
- mesh networks may not work

View File

@@ -0,0 +1,13 @@
## Remote control is not working - nothing happens
*
- Try to unpair it from your phone / computer Bluetooth settings, then re-pair it.
- Try restarting the pairing process in BikeControl
- try restarting Bluetooth on your phone and on the device you want to control
- If your other device is an iOS device, go to Settings > Accessibility > Touch > AssistiveTouch > Pointer Devices > Devices and pair your device. Make sure AssistiveTouch is enabled.
- it is very important that both devices (e.g. iPhone and iPad) receive the "pairing dialog" after initial connection. If you miss it, unpair and try again. It may take a few seconds for the dialog to appear. Afterwards you may need to click on "Reconnect" in BikeControl / restart BikeControl.
## Remote control only clicks on a single coordinate on my iPad
*
iOS seems to be buggy here - try this in the iOS settings:
AssistiveTouch settings > Pointer Devices > Devices > Connected Devices > iPhone (or BikeControl iOS) > Button 1
switch the setting to None, then back to Single-Tap and it should work again

4
INSTRUCTIONS_ROUVY.md Normal file
View File

@@ -0,0 +1,4 @@
## Local Connection method
*
The local connection method (avalable on Android, Windows and macOS) allows BikeControl to directly control Rouvy either using touch or keyboard keys. This way you don't need to select any "Controllers" at all in Rouvy.
Make sure the "Virtual Shifting Controls" are enabled: https://support.rouvy.com/hc/en-us/articles/32452137189393-Virtual-Shifting#h_01K9SWGWYMAVQV108SQ9KWQAKC

View File

@@ -1 +0,0 @@
Instructions will be added soon

0
INSTRUCTIONS_ZWIFT.md Normal file
View File

View File

@@ -1,15 +1,15 @@
# SwiftControl
# BikeControl (formerly SwiftControl)
<img src="logo.png" alt="SwiftControl Logo"/>
<img src="logo.png" alt="BikeControl Logo"/>
## Description
With SwiftControl you can **control your favorite trainer app** using your Zwift Click, Zwift Ride, Zwift Play, or other similar devices. Here's what you can do with it, depending on your configuration:
With BikeControl you can **control your favorite trainer app** using your Zwift Click, Zwift Ride, Zwift Play, Shimano Di2, or other similar devices. Here's what you can do with it, depending on your configuration:
- Virtual Gear shifting
- Steering/turning
- Steering / navigation
- adjust workout intensity
- control music on your device
- more? If you can do it via keyboard, mouse, or touch, you can do it with SwiftControl
- more? If you can do it via keyboard, mouse, or touch, you can do it with BikeControl
https://github.com/user-attachments/assets/1f81b674-1628-4763-ad66-5f3ed7a3f159
@@ -17,8 +17,8 @@ https://github.com/user-attachments/assets/1f81b674-1628-4763-ad66-5f3ed7a3f159
## Downloads
Check the compatibility matrix below!
## Download
Best follow our landing page and the "Get Started" button: [bikecontrol.app](https://bikecontrol.app/) to understand on which platform you want to run BikeControl. A testing period is available, allowing you to try out the full functionality of BikeControl:
<a href="https://play.google.com/store/apps/details?id=de.jonasbark.swiftcontrol"><img width="270" height="80" alt="GetItOnGooglePlay_Badge_Web_color_English" src="https://github.com/user-attachments/assets/a059d5a1-2efb-4f65-8117-ef6a99823b21" /></a>
@@ -30,11 +30,11 @@ Check the compatibility matrix below!
## Supported Apps
- MyWhoosh
- Zwift
- TrainingPeaks Virtual / indieVelo
- Biketerra.com
- Rouvy
- Zwift
- running SwiftControl on Android or Windows is required to act as a "Controllable" in Zwift - iOS and macOS are not able to do so
- [OpenBikeControl](https://openbikecontrol.org) compatible apps
- any other!
- You can add custom mapping and adjust touch points or keyboard shortcuts to your liking
@@ -45,57 +45,59 @@ Check the compatibility matrix below!
- Zwift Play
- Shimano Di2
- Configure your levers to use D-Fly channels with Shimano E-Tube app
- SRAM AXS/eTap
- only single or double click is supported (no individual button mapping possible, yet)
- Wahoo Kickr Bike Shift
- Wahoo Kickr Bike Pro
- CYCPLUS BC2 Virtual Shifter
- Elite Sterzo Smart (for steering support)
- Elite Square Smart Frame (beta)
- Gamepads (beta)
- Your Phone!
- Mount your phone on the handlebar to detect e.g. steering
- Available on Android and iOS
- Gamepads
- Keyboard input
- like a Companion App
- some trainers do not support keyboard input for all functions - now they do!
- useful when remapping keys from other devices using e.g. AutoHotkey
- Cheap Bluetooth buttons such as [these](https://www.amazon.com/s?k=bluetooth+remote) (beta)
- BLE HID devices and classic Bluetooth HID devices are supported
- works on Android
- on iOS and macOS requires SwiftControl to act as media player
- works out of the box on Android
- on Windows, iOS and macOS requires BikeControl to act as media player
- We're working on creating an affordable alternative based on an open standard, supported by all major trainer apps
- register your interest [here](https://openbikecontrol.org/#HARDWARE)
Support for other devices can be added; check the issues tab here on GitHub.
Support for other devices can be added; check the issues tab here on GitHub.
## Supported Accessories
- Wahoo KICKR HEADWIND (beta)
- control fan speed using your controller
## Supported Platforms
Follow this compatibility matrix. It all depends on where you want to run your trainer app (e.g. MyWhoosh on):
Follow the "Get Started" button over at [bikecontrol.app](https://bikecontrol.app) to understand on which platform you want to run BikeControl.
You can even try it out in your [Browser](https://jonasbark.github.io/swiftcontrol/), if it supports Bluetooth connections. No controlling possible, though.
| Run Trainer app (MyWhoosh, ...) on: | Possible | Link | Information |
|-------------------------------------------------------------|----------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| Android | ✅ | <a href="https://play.google.com/store/apps/details?id=de.jonasbark.swiftcontrol"><img width="270" height="80" alt="GetItOnGooglePlay_Badge_Web_color_English" src="https://github.com/user-attachments/assets/a059d5a1-2efb-4f65-8117-ef6a99823b21" /></a> | |
| iPad | ✅ | <a href="https://apps.apple.com/us/app/swiftcontrol/id6753721284?platform=iphone"><img width="270" height="80" alt="App Store" src="https://github.com/user-attachments/assets/c23f977a-48f6-4951-811e-ae530dbfa014" /></a> | You will need to use SwiftControl as a "remote" to control the trainer app on your iPad. Typically, you would use an iPhone or an Android phone for that. |
| Windows | ✅ | <a href="https://apps.microsoft.com/detail/9NP42GS03Z26"><img width="270" alt="Microsoft Store" src="https://github.com/user-attachments/assets/7a8a3cd6-ec26-4678-a850-732eedd27c48" /></a> | - Windows may flag the app as virus. It likely does so because the app controls the mouse and keyboard.<br>- Bluetooth connection unstable? You may need to use an [external Bluetooth adapter](https://github.com/jonasbark/swiftcontrol/issues/14#issuecomment-3193839509).<br>- Make sure your Zwift device is not paired with Windows Bluetooth settings: [more information](https://github.com/jonasbark/swiftcontrol/issues/70). |
| macOS | ✅ | <a href="https://apps.apple.com/us/app/swiftcontrol/id6753721284?platform=mac"><img width="270" height="80" alt="Mac App Store" src="https://github.com/user-attachments/assets/b3552436-409c-43b0-ba7d-b6a72ae30ff1" /></a> | |
| iPhone | (✅) | <a href="https://apps.apple.com/us/app/swiftcontrol/id6753721284?platform=iphone"><img width="270" height="80" alt="App Store" src="https://github.com/user-attachments/assets/c23f977a-48f6-4951-811e-ae530dbfa014" /></a> | Note that you can't run SwiftControl and your trainer app on the same iPhone due to iOS limitations, but you could use the Link method on another device to control MyWhoosh (and only MyWhoosh) on an iPhone. |
| Apple TV | (✅*) | | *only MyWhoosh using the Link method is supported - but you cannot also use MyWhoosh Link at the same time |
For testing purposes, you can also run it on [Web](https://jonasbark.github.io/swiftcontrol/), but this is just a tech demo - you won't be able to control other apps.
## Troubleshooting
## Help
Check the troubleshooting guide [here](TROUBLESHOOTING.md).
## How does it work?
The app connects to your Controller devices (such as Zwift ones) automatically. It does not connect to your trainer itself.
- **Android**: SwiftControl uses the AccessibilityService API to simulate touch gestures on specific parts of your screen to trigger actions in training apps. The service monitors which training app window is currently active to ensure gestures are sent to the correct app.
- **iOS**: use SwiftControl as a "remote control" for other devices, such as an iPad. Example scenario:
- your phone (Android/iOS) runs SwiftControl and connects to your Controller devices
- your iPad or other tablet runs e.g. MyWhoosh (does not need to have SwiftControl installed)
- If you want to use MyWhoosh, you can use the Link method to directly connect to MyWhoosh
- For other trainer apps, you need to pair SwiftControl to your iPad / tablet via Bluetooth, and your phone will send the button presses to your iPad / tablet
- **macOS** / **Windows** a keyboard or mouse click is used to trigger the action.
- There are predefined Keymaps for MyWhoosh, indieVelo / Training Peaks, and others
- You can also create your own Keymaps for any other app
- You can also use the mouse to click on a certain part of the screen, or use keyboard shortcuts
## Alternatives
- [qdomyos-zwift](https://www.qzfitness.com/) directly controls the trainer (as opposed to controlling the trainer app). This can be useful if your trainer app does not support virtual shifting.
The app connects to your Controller devices (such as Zwift ones) automatically. BikeControl uses different methods of connecting to the trainer app, depending on the trainer app and operating system:
- Connect to the trainer app on the same device or on another device using Network
- available on Android, iOS, macOS, Windows
- supported by e.g. MyWhoosh, Rouvy and Zwift
- Connect to the trainer app on another device by simulating a Bluetooth device
- available on Android, iOS, macOS, Windows
- supported by e.g. Rouvy and Zwift
- Directly control the trainer app via Accessibility features (simulating touch and keyboard input)
- available on Android, macOS, Windows
- supported by all trainer apps
- Connect to supported trainer app using the [OpenBikeControl](https://openbikecontrol.org) protocol
- available on Android, iOS, macOS, Windows
## Donate
Please consider donating to support the development of this app :)
- [via PayPal](https://paypal.me/boni)
- [via Credit Card, Google Pay, Apple Pay, etc (USD)](https://donate.stripe.com/8x24gzc5c4ZE3VJdt36J201)
- [via Credit Card, Google Pay, Apple Pay, etc (EUR)](https://donate.stripe.com/9B6aEX0muajY8bZ1Kl6J200)
- [via Credit Card, Google Pay, Apple Pay, etc. (USD)](https://donate.stripe.com/8x24gzc5c4ZE3VJdt36J201)
- [via Credit Card, Google Pay, Apple Pay, etc. (EUR)](https://donate.stripe.com/9B6aEX0muajY8bZ1Kl6J200)

View File

@@ -1,53 +1,37 @@
## Click device cannot be found
## Click / Ride device cannot be found
*
You may need to update the firmware in Zwift Companion app.
## Click device does not send any data
## Click / Ride device does not send any data
*
You may need to update the firmware in Zwift Companion app.
## My Click v2 disconnects after a minute
*
Check [this](https://github.com/jonasbark/swiftcontrol/issues/68) discussion.
To make your Click V2 work best you should connect it in the Zwift app once each day.
If you don't do that SwiftControl will need to reconnect every minute.
If you don't do that BikeControl will need to reconnect every minute.
1. Open Zwift app (not the Companion)
2. Log in (subscription not required) and open the device connection screen
3. Connect your Trainer, then connect the Click V2
4. Close the Zwift app again and connect again in SwiftControl
4. Optional: some users report that keeping the Click connected for more than a few seconds is more reliable.
5. Close the Zwift app again and connect again in BikeControl
## Android: Connection works, buttons work but nothing happens in MyWhoosh and similar
*
- especially for Redmi and other chinese Android devices please follow the instructions on [https://dontkillmyapp.com/](https://dontkillmyapp.com/):
- disable battery optimization for SwiftControl
- enable auto start of SwiftControl
- grant accessibility permission for SwiftControl
- see [https://github.com/jonasbark/swiftcontrol/issues/38](https://github.com/jonasbark/swiftcontrol/issues/38) for more details
- disable battery optimization for BikeControl
- enable auto start of BikeControl
- grant accessibility permission for BikeControl
- see [https://github.com/jonasbark/swiftcontrol/issues/38](https://github.com/OpenBikeControl/bikecontrol/issues/38) for more details
## Remote control is not working - nothing happens
- Try to unpair it from your phone / computer Bluetooth settings, then re-pair it.
- Try restarting the pairing process in SwiftControl
- try restarting Bluetooth on your phone and on the device you want to control
- If your other device is an iOS device, go to Settings > Accessibility > Touch > AssistiveTouch > Pointer Devices > Devices and pair your device. Make sure AssistiveTouch is enabled.
- it is very important that both devices (e.g. iPhone and iPad) receive the "pairing dialog" after initial connection. If you miss it, unpair and try again. It may take a few seconds for the dialog to appear. Afterwards you may need to click on "Reconnect" in SwiftControl / restart SwiftControl.
## BikeControl crashes on Windows when searching for the device
*
You're probably running into [this](https://github.com/OpenBikeControl/bikecontrol/issues/70) issue. Disconnect your controller device (e.g. Zwift Play) from Windows Bluetooth settings.
## Remote control only clicks on a single coordinate on my iPad
iOS seems to be buggy here - try this in the iOS settings:
AssistiveTouch settings > Pointer Devices > Devices > Connected Devices > iPhone (or SwiftControl iOS) > Button 1
switch the setting to None, then back to Single-Tap and it should work again
## SwiftControl crashes on Windows when searching for the device
You're probably running into [this](https://github.com/jonasbark/swiftcontrol/issues/70) issue. Disconnect your controller device (e.g. Zwift Play) from Windows Bluetooth settings.
## MyWhoosh Direct Connect never connects
The same network restrictions apply for SwiftControl as it applies to MyWhoosh Link app. Please verify with the MyWhoosh Link app if connection is possible at all.
Here are some instructions that can help:
[https://mywhoosh.com/troubleshoot/](https://mywhoosh.com/troubleshoot/)
[https://www.facebook.com/groups/mywhoosh/posts/1323791068858873/](https://www.facebook.com/groups/mywhoosh/posts/1323791068858873/)
[INSTRUCTIONS_IOS.md](INSTRUCTIONS_IOS.md)
In essence:
- your two devices (phone, tablet) need to be on the same WiFi network
- on iOS you have to turn off "Private Wi-Fi Address" in the WiFi settings
- Limit IP Address Tracking may need to be disabled
- mesh networks may not work
## My Clicks do not get recognized in MyWhoosh, but I am connected / use local control
*
Make sure you've enabled Virtual Shifting in MyWhoosh's settings

View File

@@ -1 +1 @@
3.4.0
4.1.0

View File

@@ -129,6 +129,43 @@ data class WindowEvent (
override fun hashCode(): Int = toList().hashCode()
}
/** Generated class from Pigeon that represents data sent in messages. */
data class AKeyEvent (
val source: String,
val hidKey: String,
val keyDown: Boolean,
val keyUp: Boolean
)
{
companion object {
fun fromList(pigeonVar_list: List<Any?>): AKeyEvent {
val source = pigeonVar_list[0] as String
val hidKey = pigeonVar_list[1] as String
val keyDown = pigeonVar_list[2] as Boolean
val keyUp = pigeonVar_list[3] as Boolean
return AKeyEvent(source, hidKey, keyDown, keyUp)
}
}
fun toList(): List<Any?> {
return listOf(
source,
hidKey,
keyDown,
keyUp,
)
}
override fun equals(other: Any?): Boolean {
if (other !is AKeyEvent) {
return false
}
if (this === other) {
return true
}
return AccessibilityApiPigeonUtils.deepEquals(toList(), other.toList()) }
override fun hashCode(): Int = toList().hashCode()
}
private open class AccessibilityApiPigeonCodec : StandardMessageCodec() {
override fun readValueOfType(type: Byte, buffer: ByteBuffer): Any? {
return when (type) {
@@ -142,6 +179,11 @@ private open class AccessibilityApiPigeonCodec : StandardMessageCodec() {
WindowEvent.fromList(it)
}
}
131.toByte() -> {
return (readValue(buffer) as? List<Any?>)?.let {
AKeyEvent.fromList(it)
}
}
else -> super.readValueOfType(type, buffer)
}
}
@@ -155,6 +197,10 @@ private open class AccessibilityApiPigeonCodec : StandardMessageCodec() {
stream.write(130)
writeValue(stream, value.toList())
}
is AKeyEvent -> {
stream.write(131)
writeValue(stream, value.toList())
}
else -> super.writeValue(stream, value)
}
}
@@ -168,6 +214,7 @@ interface Accessibility {
fun openPermissions()
fun performTouch(x: Double, y: Double, isKeyDown: Boolean, isKeyUp: Boolean)
fun controlMedia(action: MediaAction)
fun isRunning(): Boolean
fun ignoreHidDevices()
companion object {
@@ -249,6 +296,21 @@ interface Accessibility {
channel.setMessageHandler(null)
}
}
run {
val channel = BasicMessageChannel<Any?>(binaryMessenger, "dev.flutter.pigeon.accessibility.Accessibility.isRunning$separatedMessageChannelSuffix", codec)
if (api != null) {
channel.setMessageHandler { _, reply ->
val wrapped: List<Any?> = try {
listOf(api.isRunning())
} catch (exception: Throwable) {
AccessibilityApiPigeonUtils.wrapError(exception)
}
reply.reply(wrapped)
}
} else {
channel.setMessageHandler(null)
}
}
run {
val channel = BasicMessageChannel<Any?>(binaryMessenger, "dev.flutter.pigeon.accessibility.Accessibility.ignoreHidDevices$separatedMessageChannelSuffix", codec)
if (api != null) {
@@ -318,14 +380,14 @@ abstract class StreamEventsStreamHandler : AccessibilityApiPigeonEventChannelWra
}
}
abstract class HidKeyPressedStreamHandler : AccessibilityApiPigeonEventChannelWrapper<String> {
abstract class HidKeyPressedStreamHandler : AccessibilityApiPigeonEventChannelWrapper<AKeyEvent> {
companion object {
fun register(messenger: BinaryMessenger, streamHandler: HidKeyPressedStreamHandler, instanceName: String = "") {
var channelName: String = "dev.flutter.pigeon.accessibility.EventChannelMethods.hidKeyPressed"
if (instanceName.isNotEmpty()) {
channelName += ".$instanceName"
}
val internalStreamHandler = AccessibilityApiPigeonStreamHandler<String>(streamHandler)
val internalStreamHandler = AccessibilityApiPigeonStreamHandler<AKeyEvent>(streamHandler)
EventChannel(messenger, channelName, AccessibilityApiPigeonMethodCodec).setStreamHandler(internalStreamHandler)
}
}

View File

@@ -1,5 +1,6 @@
package de.jonasbark.accessibility
import AKeyEvent
import Accessibility
import HidKeyPressedStreamHandler
import MediaAction
@@ -51,6 +52,10 @@ class AccessibilityPlugin: FlutterPlugin, Accessibility {
return enabledServices != null && enabledServices.contains(context.packageName)
}
override fun isRunning(): Boolean {
return Observable.toService != null
}
override fun openPermissions() {
startActivity(context, Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS).apply {
flags = Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TASK
@@ -112,9 +117,9 @@ class WindowEventListener : StreamEventsStreamHandler(), Receiver {
class HidEventListener : HidKeyPressedStreamHandler(), Receiver {
private var keyEventSink: PigeonEventSink<String>? = null
private var keyEventSink: PigeonEventSink<AKeyEvent>? = null
override fun onListen(p0: Any?, sink: PigeonEventSink<String>) {
override fun onListen(p0: Any?, sink: PigeonEventSink<AKeyEvent>) {
keyEventSink = sink
}
@@ -124,6 +129,13 @@ class HidEventListener : HidKeyPressedStreamHandler(), Receiver {
override fun onKeyEvent(event: KeyEvent) {
val keyString = KeyEvent.keyCodeToString(event.keyCode)
keyEventSink?.success(keyString)
keyEventSink?.success(
AKeyEvent(
hidKey = keyString,
source = event.device.name,
keyUp = event.action == KeyEvent.ACTION_UP,
keyDown = event.action == KeyEvent.ACTION_DOWN
)
)
}
}

View File

@@ -78,9 +78,7 @@ class AccessibilityService : AccessibilityService(), Listener {
)
// Forward key events to the plugin (Flutter) and swallow them so they don't propagate.
if (event.action == KeyEvent.ACTION_DOWN) {
Observable.fromServiceKeys?.onKeyEvent(event)
}
Observable.fromServiceKeys?.onKeyEvent(event)
// Return true to indicate we've handled the event and it should be swallowed.
return true
} else {

View File

@@ -10,6 +10,8 @@ abstract class Accessibility {
void controlMedia(MediaAction action);
bool isRunning();
void ignoreHidDevices();
}
@@ -31,8 +33,17 @@ class WindowEvent {
});
}
class AKeyEvent {
final String source;
final String hidKey;
final bool keyDown;
final bool keyUp;
AKeyEvent({required this.source, required this.hidKey, required this.keyDown, required this.keyUp});
}
@EventChannelApi()
abstract class EventChannelMethods {
WindowEvent streamEvents();
String hidKeyPressed();
AKeyEvent hidKeyPressed();
}

View File

@@ -97,6 +97,62 @@ class WindowEvent {
;
}
class AKeyEvent {
AKeyEvent({
required this.source,
required this.hidKey,
required this.keyDown,
required this.keyUp,
});
String source;
String hidKey;
bool keyDown;
bool keyUp;
List<Object?> _toList() {
return <Object?>[
source,
hidKey,
keyDown,
keyUp,
];
}
Object encode() {
return _toList(); }
static AKeyEvent decode(Object result) {
result as List<Object?>;
return AKeyEvent(
source: result[0]! as String,
hidKey: result[1]! as String,
keyDown: result[2]! as bool,
keyUp: result[3]! as bool,
);
}
@override
// ignore: avoid_equals_and_hash_code_on_mutable_classes
bool operator ==(Object other) {
if (other is! AKeyEvent || other.runtimeType != runtimeType) {
return false;
}
if (identical(this, other)) {
return true;
}
return _deepEquals(encode(), other.encode());
}
@override
// ignore: avoid_equals_and_hash_code_on_mutable_classes
int get hashCode => Object.hashAll(_toList())
;
}
class _PigeonCodec extends StandardMessageCodec {
const _PigeonCodec();
@@ -111,6 +167,9 @@ class _PigeonCodec extends StandardMessageCodec {
} else if (value is WindowEvent) {
buffer.putUint8(130);
writeValue(buffer, value.encode());
} else if (value is AKeyEvent) {
buffer.putUint8(131);
writeValue(buffer, value.encode());
} else {
super.writeValue(buffer, value);
}
@@ -124,6 +183,8 @@ class _PigeonCodec extends StandardMessageCodec {
return value == null ? null : MediaAction.values[value];
case 130:
return WindowEvent.decode(readValue(buffer)!);
case 131:
return AKeyEvent.decode(readValue(buffer)!);
default:
return super.readValueOfType(type, buffer);
}
@@ -242,6 +303,34 @@ class Accessibility {
}
}
Future<bool> isRunning() async {
final String pigeonVar_channelName = 'dev.flutter.pigeon.accessibility.Accessibility.isRunning$pigeonVar_messageChannelSuffix';
final BasicMessageChannel<Object?> pigeonVar_channel = BasicMessageChannel<Object?>(
pigeonVar_channelName,
pigeonChannelCodec,
binaryMessenger: pigeonVar_binaryMessenger,
);
final Future<Object?> pigeonVar_sendFuture = pigeonVar_channel.send(null);
final List<Object?>? pigeonVar_replyList =
await pigeonVar_sendFuture as List<Object?>?;
if (pigeonVar_replyList == null) {
throw _createConnectionError(pigeonVar_channelName);
} else if (pigeonVar_replyList.length > 1) {
throw PlatformException(
code: pigeonVar_replyList[0]! as String,
message: pigeonVar_replyList[1] as String?,
details: pigeonVar_replyList[2],
);
} else if (pigeonVar_replyList[0] == null) {
throw PlatformException(
code: 'null-error',
message: 'Host platform returned null value for non-null return value.',
);
} else {
return (pigeonVar_replyList[0] as bool?)!;
}
}
Future<void> ignoreHidDevices() async {
final String pigeonVar_channelName = 'dev.flutter.pigeon.accessibility.Accessibility.ignoreHidDevices$pigeonVar_messageChannelSuffix';
final BasicMessageChannel<Object?> pigeonVar_channel = BasicMessageChannel<Object?>(
@@ -277,14 +366,14 @@ Stream<WindowEvent> streamEvents( {String instanceName = ''}) {
});
}
Stream<String> hidKeyPressed( {String instanceName = ''}) {
Stream<AKeyEvent> hidKeyPressed( {String instanceName = ''}) {
if (instanceName.isNotEmpty) {
instanceName = '.$instanceName';
}
final EventChannel hidKeyPressedChannel =
EventChannel('dev.flutter.pigeon.accessibility.EventChannelMethods.hidKeyPressed$instanceName', pigeonMethodCodec);
return hidKeyPressedChannel.receiveBroadcastStream().map((dynamic event) {
return event as String;
return event as AKeyEvent;
});
}

View File

@@ -1,7 +0,0 @@
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
<!-- The INTERNET permission is required for development. Specifically,
the Flutter tool needs it to communicate with the running application
to allow setting breakpoints, to provide hot reload, etc.
-->
<uses-permission android:name="android.permission.INTERNET"/>
</manifest>

View File

@@ -16,6 +16,7 @@
<uses-permission android:name="android.permission.FOREGROUND_SERVICE"/>
<uses-permission android:name="android.permission.FOREGROUND_SERVICE_CONNECTED_DEVICE"/>
<uses-permission android:name="android.permission.BILLING"/>
<!-- legacy for Android 9 or lower -->
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" android:maxSdkVersion="28" tools:replace="android:maxSdkVersion" />
@@ -25,7 +26,7 @@
<uses-permission android:name="android.permission.INTERNET" />
<application
android:label="SwiftControl"
android:label="BikeControl"
android:name="${applicationName}"
android:icon="@mipmap/ic_launcher">
<activity

View File

@@ -1,5 +1,6 @@
# Uncomment this line to define a global platform for your project
# platform :ios, '13.0'
platform :ios, '15.0'
# CocoaPods analytics sends network stats synchronously affecting flutter build latency.
ENV['COCOAPODS_DISABLE_STATS'] = 'true'

View File

@@ -7,12 +7,26 @@ PODS:
- Flutter (1.0.0)
- flutter_local_notifications (0.0.1):
- Flutter
- flutter_secure_storage_darwin (10.0.0):
- Flutter
- FlutterMacOS
- gamepads_ios (0.1.1):
- Flutter
- image_picker_ios (0.0.1):
- Flutter
- in_app_purchase_storekit (0.0.1):
- Flutter
- FlutterMacOS
- in_app_review (2.0.0):
- Flutter
- integration_test (0.0.1):
- Flutter
- ios_receipt (0.0.1):
- Flutter
- media_key_detector_ios (0.0.1):
- Flutter
- nsd_ios (0.0.1):
- Flutter
- package_info_plus (0.4.5):
- Flutter
- path_provider_foundation (0.0.1):
@@ -22,6 +36,8 @@ PODS:
- Flutter
- restart_app (0.0.1):
- Flutter
- sensors_plus (0.0.1):
- Flutter
- shared_preferences_foundation (0.0.1):
- Flutter
- FlutterMacOS
@@ -38,13 +54,20 @@ DEPENDENCIES:
- device_info_plus (from `.symlinks/plugins/device_info_plus/ios`)
- Flutter (from `Flutter`)
- flutter_local_notifications (from `.symlinks/plugins/flutter_local_notifications/ios`)
- flutter_secure_storage_darwin (from `.symlinks/plugins/flutter_secure_storage_darwin/darwin`)
- gamepads_ios (from `.symlinks/plugins/gamepads_ios/ios`)
- image_picker_ios (from `.symlinks/plugins/image_picker_ios/ios`)
- in_app_purchase_storekit (from `.symlinks/plugins/in_app_purchase_storekit/darwin`)
- in_app_review (from `.symlinks/plugins/in_app_review/ios`)
- integration_test (from `.symlinks/plugins/integration_test/ios`)
- ios_receipt (from `.symlinks/plugins/ios_receipt/ios`)
- media_key_detector_ios (from `.symlinks/plugins/media_key_detector_ios/ios`)
- nsd_ios (from `.symlinks/plugins/nsd_ios/ios`)
- package_info_plus (from `.symlinks/plugins/package_info_plus/ios`)
- path_provider_foundation (from `.symlinks/plugins/path_provider_foundation/darwin`)
- permission_handler_apple (from `.symlinks/plugins/permission_handler_apple/ios`)
- restart_app (from `.symlinks/plugins/restart_app/ios`)
- sensors_plus (from `.symlinks/plugins/sensors_plus/ios`)
- shared_preferences_foundation (from `.symlinks/plugins/shared_preferences_foundation/darwin`)
- universal_ble (from `.symlinks/plugins/universal_ble/darwin`)
- url_launcher_ios (from `.symlinks/plugins/url_launcher_ios/ios`)
@@ -59,12 +82,24 @@ EXTERNAL SOURCES:
:path: Flutter
flutter_local_notifications:
:path: ".symlinks/plugins/flutter_local_notifications/ios"
flutter_secure_storage_darwin:
:path: ".symlinks/plugins/flutter_secure_storage_darwin/darwin"
gamepads_ios:
:path: ".symlinks/plugins/gamepads_ios/ios"
image_picker_ios:
:path: ".symlinks/plugins/image_picker_ios/ios"
in_app_purchase_storekit:
:path: ".symlinks/plugins/in_app_purchase_storekit/darwin"
in_app_review:
:path: ".symlinks/plugins/in_app_review/ios"
integration_test:
:path: ".symlinks/plugins/integration_test/ios"
ios_receipt:
:path: ".symlinks/plugins/ios_receipt/ios"
media_key_detector_ios:
:path: ".symlinks/plugins/media_key_detector_ios/ios"
nsd_ios:
:path: ".symlinks/plugins/nsd_ios/ios"
package_info_plus:
:path: ".symlinks/plugins/package_info_plus/ios"
path_provider_foundation:
@@ -73,6 +108,8 @@ EXTERNAL SOURCES:
:path: ".symlinks/plugins/permission_handler_apple/ios"
restart_app:
:path: ".symlinks/plugins/restart_app/ios"
sensors_plus:
:path: ".symlinks/plugins/sensors_plus/ios"
shared_preferences_foundation:
:path: ".symlinks/plugins/shared_preferences_foundation/darwin"
universal_ble:
@@ -87,18 +124,25 @@ SPEC CHECKSUMS:
device_info_plus: bf2e3232933866d73fe290f2942f2156cdd10342
Flutter: cabc95a1d2626b1b06e7179b784ebcf0c0cde467
flutter_local_notifications: ff50f8405aaa0ccdc7dcfb9022ca192e8ad9688f
flutter_secure_storage_darwin: 557817588b80e60213cbecb573c45c76b788018d
gamepads_ios: 1d2930c7a4450a9a1b57444ebf305a6a6cbeea0b
image_picker_ios: c560581cceedb403a6ff17f2f816d7fea1421fc1
in_app_purchase_storekit: 2342c0a5da86593124d08dd13d920f39a52b273a
in_app_review: 436034b18594851a7328d7f1c2ed5ec235b79cfc
integration_test: 252f60fa39af5e17c3aa9899d35d908a0721b573
ios_receipt: c2d5b4c36953c377a024992393976214ce6951e6
media_key_detector_ios: 7ff9aefdfea00bb7b71e184132381b7d0e7e1269
nsd_ios: 8c37babdc6538e3350dbed3a52674d2edde98173
package_info_plus: c0502532a26c7662a62a356cebe2692ec5fe4ec4
path_provider_foundation: 0b743cbb62d8e47eab856f09262bb8c1ddcfe6ba
permission_handler_apple: 9878588469a2b0d0fc1e048d9f43605f92e6cec2
restart_app: 806659942bf932f6ce51c5372f91ce5e81c8c14a
sensors_plus: 7229095999f30740798f0eeef5cd120357a8f4f2
shared_preferences_foundation: fcdcbc04712aee1108ac7fda236f363274528f78
universal_ble: cf52a7b3fd2e7c14d6d7262e9fdadb72ab6b88a6
url_launcher_ios: 5334b05cef931de560670eeae103fd3e431ac3fe
wakelock_plus: 76957ab028e12bfa4e66813c99e46637f367fc7e
PODFILE CHECKSUM: 3c63482e143d1b91d2d2560aee9fb04ecc74ac7e
PODFILE CHECKSUM: 7ebd5c9b932b3af79d5c67e3af873118b74e970f
COCOAPODS: 1.16.2

View File

@@ -65,6 +65,7 @@
97C147021CF9000F007C117D /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; };
DFFDC4B9C4D6EF6A3BDE2E73 /* Pods-RunnerTests.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.debug.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.debug.xcconfig"; sourceTree = "<group>"; };
EFDECED99A47773C293F8819 /* Pods-RunnerTests.profile.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.profile.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.profile.xcconfig"; sourceTree = "<group>"; };
F0D040E82EEF2560009B19C0 /* Runner.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = Runner.entitlements; sourceTree = "<group>"; };
/* End PBXFileReference section */
/* Begin PBXFrameworksBuildPhase section */
@@ -152,6 +153,7 @@
97C146F01CF9000F007C117D /* Runner */ = {
isa = PBXGroup;
children = (
F0D040E82EEF2560009B19C0 /* Runner.entitlements */,
97C146FA1CF9000F007C117D /* Main.storyboard */,
97C146FD1CF9000F007C117D /* Assets.xcassets */,
97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */,
@@ -487,12 +489,14 @@
buildSettings = {
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
CLANG_ENABLE_MODULES = YES;
CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements;
CODE_SIGN_IDENTITY = "Apple Development";
CODE_SIGN_STYLE = Automatic;
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
DEVELOPMENT_TEAM = UZRHKPVWN9;
ENABLE_BITCODE = NO;
INFOPLIST_FILE = Runner/Info.plist;
IPHONEOS_DEPLOYMENT_TARGET = 15.6;
LD_RUNPATH_SEARCH_PATHS = (
"$(inherited)",
"@executable_path/Frameworks",
@@ -673,12 +677,14 @@
buildSettings = {
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
CLANG_ENABLE_MODULES = YES;
CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements;
CODE_SIGN_IDENTITY = "Apple Development";
CODE_SIGN_STYLE = Automatic;
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
DEVELOPMENT_TEAM = UZRHKPVWN9;
ENABLE_BITCODE = NO;
INFOPLIST_FILE = Runner/Info.plist;
IPHONEOS_DEPLOYMENT_TARGET = 15.6;
LD_RUNPATH_SEARCH_PATHS = (
"$(inherited)",
"@executable_path/Frameworks",
@@ -699,12 +705,14 @@
buildSettings = {
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
CLANG_ENABLE_MODULES = YES;
CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements;
CODE_SIGN_IDENTITY = "Apple Development";
CODE_SIGN_STYLE = Automatic;
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
DEVELOPMENT_TEAM = UZRHKPVWN9;
ENABLE_BITCODE = NO;
INFOPLIST_FILE = Runner/Info.plist;
IPHONEOS_DEPLOYMENT_TARGET = 15.6;
LD_RUNPATH_SEARCH_PATHS = (
"$(inherited)",
"@executable_path/Frameworks",

View File

@@ -8,6 +8,8 @@ import UIKit
didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?
) -> Bool {
GeneratedPluginRegistrant.register(with: self)
UNUserNotificationCenter.current().delegate = self as? UNUserNotificationCenterDelegate
return super.application(application, didFinishLaunchingWithOptions: launchOptions)
}
}

View File

@@ -7,7 +7,7 @@
<key>CFBundleDevelopmentRegion</key>
<string>$(DEVELOPMENT_LANGUAGE)</string>
<key>CFBundleDisplayName</key>
<string>SwiftControl</string>
<string>BikeControl</string>
<key>CFBundleExecutable</key>
<string>$(EXECUTABLE_NAME)</string>
<key>CFBundleIdentifier</key>
@@ -29,16 +29,22 @@
<key>LSRequiresIPhoneOS</key>
<true/>
<key>NSBluetoothAlwaysUsageDescription</key>
<string>SwiftControl uses Bluetooth to connect to accessories.</string>
<string>BikeControl uses Bluetooth to connect to accessories.</string>
<key>NSLocalNetworkUsageDescription</key>
<string>This app connects to your trainer app on your local network.</string>
<key>NSMotionUsageDescription</key>
<string>Access your accelerometer and gyroscope for steering support via your phone.</string>
<key>NSBonjourServices</key>
<array>
<string>_wahoo-fitness-tnp._tcp</string>
<string>_openbikecontrol._tcp</string>
</array>
<key>UIApplicationSupportsIndirectInputEvents</key>
<true/>
<key>UIBackgroundModes</key>
<array>
<string>bluetooth-peripheral</string>
<string>bluetooth-central</string>
<string>audio</string>
</array>
<key>UILaunchStoryboardName</key>
<string>LaunchScreen</string>

View File

@@ -0,0 +1,8 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>keychain-access-groups</key>
<array/>
</dict>
</plist>

2
ios_receipt/.gitattributes vendored Normal file
View File

@@ -0,0 +1,2 @@
# Auto detect text files and perform LF normalization
* text=auto

30
ios_receipt/.gitignore vendored Normal file
View File

@@ -0,0 +1,30 @@
# Miscellaneous
*.class
*.log
*.pyc
*.swp
.DS_Store
.atom/
.buildlog/
.history
.svn/
migrate_working_dir/
# IntelliJ related
*.iml
*.ipr
*.iws
.idea/
# The .vscode folder contains launch configuration and tasks you configure in
# VS Code which you may wish to be included in version control, so this line
# is commented out by default.
#.vscode/
# Flutter/Dart/Pub related
# Libraries should not include pubspec.lock, per https://dart.dev/guides/libraries/private-files#pubspeclock.
/pubspec.lock
**/doc/api/
.dart_tool/
.packages
build/

30
ios_receipt/.metadata Normal file
View File

@@ -0,0 +1,30 @@
# This file tracks properties of this Flutter project.
# Used by Flutter tool to assess capabilities and perform upgrades etc.
#
# This file should be version controlled and should not be manually edited.
version:
revision: "367f9ea16bfae1ca451b9cc27c1366870b187ae2"
channel: "stable"
project_type: plugin
# Tracks metadata for the flutter migrate command
migration:
platforms:
- platform: root
create_revision: 367f9ea16bfae1ca451b9cc27c1366870b187ae2
base_revision: 367f9ea16bfae1ca451b9cc27c1366870b187ae2
- platform: ios
create_revision: 367f9ea16bfae1ca451b9cc27c1366870b187ae2
base_revision: 367f9ea16bfae1ca451b9cc27c1366870b187ae2
# User provided section
# List of Local paths (relative to this file) that should be
# ignored by the migrate tool.
#
# Files that are not part of the templates will be ignored by default.
unmanaged_files:
- 'lib/main.dart'
- 'ios/Runner.xcodeproj/project.pbxproj'

7
ios_receipt/CHANGELOG.md Normal file
View File

@@ -0,0 +1,7 @@
## 1.1.0
* Migrate to StoreKit2
## 1.0.0
* Implement main method

21
ios_receipt/LICENSE Normal file
View File

@@ -0,0 +1,21 @@
MIT License
Copyright (c) 2023 Dmytro O. Kut'ko
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.

32
ios_receipt/README.md Normal file
View File

@@ -0,0 +1,32 @@
# IosReceipt
The IosReceipt package allows you to easily fetch the App Store receipt in your Flutter application on the iOS platform.
## Installation
Add the following dependency to your `pubspec.yaml` file:
```yaml
dependencies:
ios_receipt: ^0.0.1 # Use the latest version of the package
```
Then, run flutter pub get to install the package.
## Usage
Method for get transactions with StoreKit2
```dart
final transactions = await IosReceipt.getAllTransactions();
```
## Based on
This package is based on the [appStoreReceiptURL](https://developer.apple.com/documentation/foundation/nsbundle/1407276-appstorereceipturl) from the official Apple documentation.
## Note
- The receipt isn't necessary if you use AppTransaction to validate the app download, or Transaction to validate in-app purchases.
- If the receipt is invalid or missing in your app, use SKReceiptRefreshRequest to request a new receipt.
## Testing Environments
Keep in mind that receipts aren't initially present in iOS and iPadOS apps in the sandbox environment and in Xcode. Apps receive a receipt after the tester completes the first in-app purchase.
## License
This project is licensed under the MIT License.

View File

@@ -0,0 +1,4 @@
include: package:flutter_lints/flutter.yaml
# Additional information about this file can be found at
# https://dart.dev/guides/language/analysis-options

38
ios_receipt/ios/.gitignore vendored Normal file
View File

@@ -0,0 +1,38 @@
.idea/
.vagrant/
.sconsign.dblite
.svn/
.DS_Store
*.swp
profile
DerivedData/
build/
GeneratedPluginRegistrant.h
GeneratedPluginRegistrant.m
.generated/
*.pbxuser
*.mode1v3
*.mode2v3
*.perspectivev3
!default.pbxuser
!default.mode1v3
!default.mode2v3
!default.perspectivev3
xcuserdata
*.moved-aside
*.pyc
*sync/
Icon?
.tags*
/Flutter/Generated.xcconfig
/Flutter/ephemeral/
/Flutter/flutter_export_environment.sh

View File

View File

@@ -0,0 +1,73 @@
import Flutter
import UIKit
import StoreKit
public class IosReceiptPlugin: NSObject, FlutterPlugin {
private func getAppleReceipt() -> String? {
guard let url = Bundle.main.appStoreReceiptURL,
FileManager.default.fileExists(atPath: url.path) else {
return nil
}
let data = try? Data(contentsOf: url, options: .alwaysMapped)
return data?.base64EncodedString()
}
private func isSandbox() -> Bool {
guard let path = Bundle.main.appStoreReceiptURL?.path else {
return false
}
return path.contains("CoreSimulator") || path.contains("sandboxReceipt")
}
private func getAllTransactions() async -> [[String: Any]] {
var list: [[String: Any]] = []
if #available(iOS 15.0, *) {
for await result in Transaction.all {
switch result {
case .verified(let tx):
var item: [String: Any] = [
"productId": tx.productID,
"transactionId": String(tx.id),
"originalTransactionId": String(tx.originalID),
"purchaseDate": ISO8601DateFormatter().string(from: tx.purchaseDate)
]
if let revocationDate = tx.revocationDate {
item["revocationDate"] = ISO8601DateFormatter().string(from: revocationDate)
}
if let reason = tx.revocationReason {
item["revocationReason"] = reason.rawValue
}
if #available(iOS 16.0, *) {
item["jws"] = result.jwsRepresentation
}
list.append(item)
case .unverified(_, _):
continue
}
}
}
return list
}
public static func register(with registrar: FlutterPluginRegistrar) {
let channel = FlutterMethodChannel(name: "ios_receipt", binaryMessenger: registrar.messenger())
let instance = IosReceiptPlugin()
registrar.addMethodCallDelegate(instance, channel: channel)
}
public func handle(_ call: FlutterMethodCall, result: @escaping FlutterResult) {
switch call.method {
case "getAppleReceipt":
result(getAppleReceipt())
case "isSandbox":
result(isSandbox())
case "getAllTransactions":
Task { result(await self.getAllTransactions()) }
default:
result(FlutterMethodNotImplemented)
}
}
}

View File

@@ -0,0 +1,23 @@
#
# To learn more about a Podspec see http://guides.cocoapods.org/syntax/podspec.html.
# Run `pod lib lint ios_receipt.podspec` to validate before publishing.
#
Pod::Spec.new do |s|
s.name = 'ios_receipt'
s.version = '0.0.1'
s.summary = 'A new Flutter plugin project.'
s.description = <<-DESC
A new Flutter plugin project.
DESC
s.homepage = 'http://example.com'
s.license = { :file => '../LICENSE' }
s.author = { 'Your Company' => 'email@example.com' }
s.source = { :path => '.' }
s.source_files = 'Classes/**/*'
s.dependency 'Flutter'
s.platform = :ios, '15.0'
# Flutter.framework does not contain a i386 slice.
s.pod_target_xcconfig = { 'DEFINES_MODULE' => 'YES', 'EXCLUDED_ARCHS[sdk=iphonesimulator*]' => 'i386' }
s.swift_version = '5.0'
end

View File

@@ -0,0 +1,22 @@
import 'package:ios_receipt/models/transaction.dart';
import 'ios_receipt_platform_interface.dart';
class IosReceipt {
static Future<String?> getAppleReceipt() {
return IosReceiptPlatform.instance.getAppleReceipt();
}
static Future<bool> isSandbox() {
return IosReceiptPlatform.instance.isSandbox();
}
static Future<List<Transaction>> getAllTransactions() async {
final list = await IosReceiptPlatform.instance.getAllTransactions();
final result = <Transaction>[];
for (var data in list) {
result.add(Transaction.fromMap(data));
}
return result..sort((a, b) => a.purchaseDate.compareTo(b.purchaseDate));
}
}

View File

@@ -0,0 +1,29 @@
import 'package:flutter/foundation.dart';
import 'package:flutter/services.dart';
import 'ios_receipt_platform_interface.dart';
/// An implementation of [IosReceiptPlatform] that uses method channels.
class MethodChannelIosReceipt extends IosReceiptPlatform {
/// The method channel used to interact with the native platform.
@visibleForTesting
final methodChannel = const MethodChannel('ios_receipt');
@override
Future<String?> getAppleReceipt() async {
final version = await methodChannel.invokeMethod<String>('getAppleReceipt');
return version;
}
@override
Future<List<Map<String, dynamic>>> getAllTransactions() async {
final list = await methodChannel.invokeMethod('getAllTransactions');
return (list as List).map((e) => Map<String, dynamic>.from(e)).toList();
}
@override
Future<bool> isSandbox() async {
final isSandbox = await methodChannel.invokeMethod<bool>('isSandbox');
return isSandbox ?? false;
}
}

View File

@@ -0,0 +1,37 @@
import 'package:plugin_platform_interface/plugin_platform_interface.dart';
import 'ios_receipt_method_channel.dart';
abstract class IosReceiptPlatform extends PlatformInterface {
/// Constructs a IosReceiptPlatform.
IosReceiptPlatform() : super(token: _token);
static final Object _token = Object();
static IosReceiptPlatform _instance = MethodChannelIosReceipt();
/// The default instance of [IosReceiptPlatform] to use.
///
/// Defaults to [MethodChannelIosReceipt].
static IosReceiptPlatform get instance => _instance;
/// Platform-specific implementations should set this with their own
/// platform-specific class that extends [IosReceiptPlatform] when
/// they register themselves.
static set instance(IosReceiptPlatform instance) {
PlatformInterface.verifyToken(instance, _token);
_instance = instance;
}
Future<String?> getAppleReceipt() {
throw UnimplementedError('platformVersion() has not been implemented.');
}
Future<List<Map<String, dynamic>>> getAllTransactions() {
throw UnimplementedError('getAllTransactions() has not been implemented.');
}
Future<bool> isSandbox() async {
throw UnimplementedError('isSandbox() has not been implemented.');
}
}

View File

@@ -0,0 +1,41 @@
class Transaction {
const Transaction({
required this.jws,
required this.productId,
required this.transactionId,
required this.purchaseDate,
required this.originalTransactionId,
});
final String? jws;
final String productId;
final String transactionId;
final DateTime purchaseDate;
final String originalTransactionId;
@override
bool operator ==(covariant Transaction other) {
if (identical(this, other)) return true;
return other.jws == jws &&
other.productId == productId &&
other.transactionId == transactionId &&
other.purchaseDate == purchaseDate &&
other.originalTransactionId == originalTransactionId;
}
@override
int get hashCode =>
jws.hashCode ^
productId.hashCode ^
transactionId.hashCode ^
purchaseDate.hashCode ^
originalTransactionId.hashCode;
factory Transaction.fromMap(Map<String, dynamic> map) => Transaction(
jws: map['jws'] as String?,
productId: map['productId'] as String,
transactionId: map['transactionId'] as String,
purchaseDate: DateTime.parse(map['purchaseDate'] as String),
originalTransactionId: map['originalTransactionId'] as String,
);
}

38
ios_receipt/macos/.gitignore vendored Normal file
View File

@@ -0,0 +1,38 @@
.idea/
.vagrant/
.sconsign.dblite
.svn/
.DS_Store
*.swp
profile
DerivedData/
build/
GeneratedPluginRegistrant.h
GeneratedPluginRegistrant.m
.generated/
*.pbxuser
*.mode1v3
*.mode2v3
*.perspectivev3
!default.pbxuser
!default.mode1v3
!default.mode2v3
!default.perspectivev3
xcuserdata
*.moved-aside
*.pyc
*sync/
Icon?
.tags*
/Flutter/Generated.xcconfig
/Flutter/ephemeral/
/Flutter/flutter_export_environment.sh

View File

View File

@@ -0,0 +1,73 @@
import FlutterMacOS
import StoreKit
public class IosReceiptPlugin: NSObject, FlutterPlugin {
private func isSandbox() -> Bool {
guard let path = Bundle.main.appStoreReceiptURL?.path else {
return false
}
return path.contains("CoreSimulator") || path.contains("sandboxReceipt")
}
private func getAppleReceipt() -> String? {
guard let url = Bundle.main.appStoreReceiptURL,
FileManager.default.fileExists(atPath: url.path) else {
return nil
}
let data = try? Data(contentsOf: url, options: .alwaysMapped)
return data?.base64EncodedString()
}
private func getAllTransactions() async -> [[String: Any]] {
var list: [[String: Any]] = []
if #available(iOS 15.0, *) {
for await result in Transaction.all {
switch result {
case .verified(let tx):
var item: [String: Any] = [
"productId": tx.productID,
"transactionId": String(tx.id),
"originalTransactionId": String(tx.originalID),
"purchaseDate": ISO8601DateFormatter().string(from: tx.purchaseDate)
]
if let revocationDate = tx.revocationDate {
item["revocationDate"] = ISO8601DateFormatter().string(from: revocationDate)
}
if let reason = tx.revocationReason {
item["revocationReason"] = reason.rawValue
}
if #available(iOS 16.0, *) {
item["jws"] = result.jwsRepresentation
}
list.append(item)
case .unverified(_, _):
continue
}
}
}
return list
}
public static func register(with registrar: FlutterPluginRegistrar) {
let channel = FlutterMethodChannel(name: "ios_receipt", binaryMessenger: registrar.messenger)
let instance = IosReceiptPlugin()
registrar.addMethodCallDelegate(instance, channel: channel)
}
public func handle(_ call: FlutterMethodCall, result: @escaping FlutterResult) {
switch call.method {
case "getAppleReceipt":
result(getAppleReceipt())
case "isSandbox":
result(isSandbox())
case "getAllTransactions":
Task { result(await self.getAllTransactions()) }
default:
result(FlutterMethodNotImplemented)
}
}
}

View File

@@ -0,0 +1,23 @@
#
# To learn more about a Podspec see http://guides.cocoapods.org/syntax/podspec.html.
# Run `pod lib lint ios_receipt.podspec` to validate before publishing.
#
Pod::Spec.new do |s|
s.name = 'ios_receipt'
s.version = '0.0.1'
s.summary = 'A new Flutter plugin project.'
s.description = <<-DESC
A new Flutter plugin project.
DESC
s.homepage = 'http://example.com'
s.license = { :file => '../LICENSE' }
s.author = { 'Your Company' => 'email@example.com' }
s.source = { :path => '.' }
s.source_files = 'Classes/**/*'
s.dependency 'FlutterMacOS'
s.platform = :osx, '12.00'
# Flutter.framework does not contain a i386 slice.
s.pod_target_xcconfig = { 'DEFINES_MODULE' => 'YES', 'EXCLUDED_ARCHS[sdk=iphonesimulator*]' => 'i386' }
s.swift_version = '5.0'
end

26
ios_receipt/pubspec.yaml Normal file
View File

@@ -0,0 +1,26 @@
name: ios_receipt
description: The IosReceipt package allows you to easily fetch the App Store receipt in your Flutter application on the iOS platform.
version: 1.1.0
homepage: https://github.com/DimaKutko/ios_receipt
environment:
sdk: ">=3.8.0 <4.0.0"
flutter: ">=3.32.0"
dependencies:
flutter:
sdk: flutter
plugin_platform_interface: ^2.1.8
dev_dependencies:
flutter_test:
sdk: flutter
flutter_lints: ^6.0.0
flutter:
plugin:
platforms:
ios:
pluginClass: IosReceiptPlugin
macos:
pluginClass: IosReceiptPlugin

View File

@@ -1,25 +1,25 @@
import 'dart:async';
import 'dart:io';
import 'dart:ui';
import 'package:bike_control/bluetooth/devices/bluetooth_device.dart';
import 'package:bike_control/bluetooth/devices/gamepad/gamepad_device.dart';
import 'package:bike_control/bluetooth/devices/gyroscope/gyroscope_steering.dart';
import 'package:bike_control/bluetooth/devices/hid/hid_device.dart';
import 'package:bike_control/bluetooth/devices/wahoo/wahoo_kickr_headwind.dart';
import 'package:bike_control/bluetooth/devices/zwift/ftms_mdns_emulator.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zp.pb.dart';
import 'package:bike_control/gen/l10n.dart';
import 'package:bike_control/main.dart';
import 'package:bike_control/utils/actions/android.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/requirements/android.dart';
import 'package:dartx/dartx.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter_local_notifications/flutter_local_notifications.dart';
import 'package:gamepads/gamepads.dart';
import 'package:media_key_detector/media_key_detector.dart';
import 'package:swift_control/bluetooth/devices/bluetooth_device.dart';
import 'package:swift_control/bluetooth/devices/gamepad/gamepad_device.dart';
import 'package:swift_control/bluetooth/devices/hid/hid_device.dart';
import 'package:swift_control/main.dart';
import 'package:swift_control/utils/actions/android.dart';
import 'package:swift_control/utils/keymap/buttons.dart';
import 'package:swift_control/utils/keymap/keymap.dart';
import 'package:swift_control/utils/requirements/android.dart';
import 'package:universal_ble/universal_ble.dart';
import '../utils/keymap/apps/my_whoosh.dart';
import 'devices/base_device.dart';
import 'devices/link/link_device.dart';
import 'devices/zwift/constants.dart';
import 'messages/notification.dart';
@@ -28,9 +28,14 @@ class Connection {
List<BluetoothDevice> get bluetoothDevices => devices.whereType<BluetoothDevice>().toList();
List<GamepadDevice> get gamepadDevices => devices.whereType<GamepadDevice>().toList();
List<BaseDevice> get controllerDevices => [...bluetoothDevices, ...gamepadDevices, ...devices.whereType<HidDevice>()];
List<BaseDevice> get remoteDevices =>
devices.whereNot((d) => d is BluetoothDevice || d is GamepadDevice || d is HidDevice).toList();
List<GyroscopeSteering> get gyroscopeDevices => devices.whereType<GyroscopeSteering>().toList();
List<WahooKickrHeadwind> get accessories => devices.whereType<WahooKickrHeadwind>().toList();
List<BaseDevice> get controllerDevices => [
...bluetoothDevices.where((d) => d is! WahooKickrHeadwind),
...gamepadDevices,
...gyroscopeDevices,
...devices.whereType<HidDevice>(),
];
var _androidNotificationsSetup = false;
@@ -40,6 +45,7 @@ class Connection {
final Map<BaseDevice, StreamSubscription<BaseNotification>> _streamSubscriptions = {};
final StreamController<BaseNotification> _actionStreams = StreamController<BaseNotification>.broadcast();
Stream<BaseNotification> get actionStream => _actionStreams.stream;
List<({DateTime date, String entry})> lastLogEntries = [];
final Map<BaseDevice, StreamSubscription<bool>> _connectionSubscriptions = {};
final StreamController<BaseDevice> _connectionStreams = StreamController<BaseDevice>.broadcast();
@@ -48,27 +54,23 @@ class Connection {
final _lastScanResult = <BleDevice>[];
final ValueNotifier<bool> hasDevices = ValueNotifier(false);
final ValueNotifier<bool> isScanning = ValueNotifier(false);
final ValueNotifier<bool> isMediaKeyDetectionEnabled = ValueNotifier(false);
Timer? _gamePadSearchTimer;
final _dontAllowReconnectDevices = <String>{};
void initialize() {
isMediaKeyDetectionEnabled.addListener(() {
if (!isMediaKeyDetectionEnabled.value) {
mediaKeyDetector.setIsPlaying(isPlaying: false);
mediaKeyDetector.removeListener(_onMediaKeyDetectedListener);
} else {
mediaKeyDetector.addListener(_onMediaKeyDetectedListener);
mediaKeyDetector.setIsPlaying(isPlaying: true);
}
actionStream.listen((log) {
lastLogEntries.add((date: DateTime.now(), entry: log.toString()));
lastLogEntries = lastLogEntries.takeLast(kIsWeb ? 1000 : 60).toList();
});
UniversalBle.onAvailabilityChange = (available) {
_actionStreams.add(BluetoothAvailabilityNotification(available == AvailabilityState.poweredOn));
if (available == AvailabilityState.poweredOn && !kIsWeb) {
performScanning();
core.permissions.getScanRequirements().then((perms) {
if (perms.isEmpty) {
performScanning();
}
});
} else if (available == AvailabilityState.poweredOff) {
reset();
}
@@ -87,13 +89,13 @@ class Connection {
_lastScanResult.add(result);
if (kDebugMode) {
print('Scan result: ${result.name} - ${result.deviceId}');
debugPrint('Scan result: ${result.name} - ${result.deviceId}');
}
final scanResult = BluetoothDevice.fromScanResult(result);
if (scanResult != null) {
_actionStreams.add(LogNotification('Found new device: ${scanResult.runtimeType}'));
_actionStreams.add(LogNotification('Found new device: ${kIsWeb ? scanResult.name : scanResult.runtimeType}'));
addDevices([scanResult]);
} else {
final manufacturerData = result.manufacturerDataList;
@@ -101,20 +103,42 @@ class Connection {
.firstOrNullWhere((e) => e.companyId == ZwiftConstants.ZWIFT_MANUFACTURER_ID)
?.payload;
if (data != null && kDebugMode) {
_actionStreams.add(LogNotification('Found unknown device with identifier: ${data.firstOrNull}'));
_actionStreams.add(
LogNotification('Found unknown device ${result.name} with identifier: ${data.firstOrNull}'),
);
}
}
}
};
UniversalBle.onValueChange = (deviceId, characteristicUuid, value) {
UniversalBle.onValueChange = (deviceId, characteristicUuid, value) async {
final device = bluetoothDevices.firstOrNullWhere((e) => e.device.deviceId == deviceId);
if (device == null) {
_actionStreams.add(LogNotification('Device not found: $deviceId'));
UniversalBle.disconnect(deviceId);
return;
} else {
device.processCharacteristic(characteristicUuid, value);
if (kIsWeb) {
// on web, log all characteristic changes for debugging
_actionStreams.add(
LogNotification(
'Characteristic update for device ${device.name}, char: $characteristicUuid, value: ${bytesToReadableHex(value)}',
),
);
}
try {
await device.processCharacteristic(characteristicUuid, value);
} catch (e, backtrace) {
_actionStreams.add(
LogNotification(
"Error processing characteristic for device ${device.name} and char: $characteristicUuid: $e\n$backtrace",
),
);
if (kDebugMode) {
print(e);
print("backtrace: $backtrace");
}
}
}
};
@@ -125,6 +149,17 @@ class Connection {
_lastScanResult.removeWhere((d) => d.deviceId == deviceId);
}
};
if (!kIsWeb && !screenshotMode) {
core.permissions.getScanRequirements().then((perms) {
if (perms.isEmpty) {
performScanning();
}
});
if (core.settings.getPhoneSteeringEnabled()) {
toggleGyroscopeSteering(true);
}
}
}
Future<void> performScanning() async {
@@ -134,6 +169,10 @@ class Connection {
isScanning.value = true;
_actionStreams.add(LogNotification('Scanning for devices...'));
if (screenshotMode) {
return;
}
// does not work on web, may not work on Windows
if (!kIsWeb && !Platform.isWindows) {
UniversalBle.getSystemDevices(
@@ -155,7 +194,7 @@ class Connection {
if (!kIsWeb) {
_gamePadSearchTimer = Timer.periodic(Duration(seconds: 3), (_) {
Gamepads.list().then((list) {
final pads = list.map((pad) => GamepadDevice(pad.name, id: pad.id)).toList();
final pads = list.map((pad) => GamepadDevice(pad.name.isEmpty ? 'Gamepad' : pad.name, id: pad.id)).toList();
addDevices(pads);
final removedDevices = gamepadDevices.where((device) => list.none((pad) => pad.id == device.id)).toList();
@@ -169,78 +208,93 @@ class Connection {
}
});
});
Gamepads.list().then((list) {
final pads = list.map((pad) => GamepadDevice(pad.name, id: pad.id)).toList();
addDevices(pads);
});
} else {
isScanning.value = false;
}
if (settings.getMyWhooshLinkEnabled() && settings.getTrainerApp() is MyWhoosh && !whooshLink.isStarted.value) {
startMyWhooshServer();
if (devices.isNotEmpty && !_androidNotificationsSetup && !kIsWeb && Platform.isAndroid) {
_androidNotificationsSetup = true;
// start foreground service only when app is in foreground
NotificationRequirement.addPersistentNotification().catchError((e) {
_actionStreams.add(LogNotification(e.toString()));
});
}
}
Future<void> startMyWhooshServer() {
return whooshLink.startServer(
onConnected: (socket) {
final existing = remoteDevices.firstOrNullWhere(
(e) => e is LinkDevice && e.identifier == socket.remoteAddress.address,
);
if (existing != null) {
existing.isConnected = true;
signalChange(existing);
}
},
onDisconnected: (socket) {
final device = devices.firstOrNullWhere(
(device) => device is LinkDevice && device.identifier == socket.remoteAddress.address,
);
if (device != null) {
devices.remove(device);
signalChange(device);
}
},
);
return core.whooshLink.startServer().catchError((e) {
core.settings.setMyWhooshLinkEnabled(false);
_actionStreams.add(LogNotification('Error starting MyWhoosh "Link" server: $e'));
_actionStreams.add(
AlertNotification(
LogLevel.LOGLEVEL_ERROR,
'Error starting MyWhoosh "Link" server. Please make sure the "MyWhoosh Link" app is not already running on this device.',
),
);
});
}
void addDevices(List<BaseDevice> dev) {
final newDevices = dev
.where((device) => !devices.contains(device) && !_dontAllowReconnectDevices.contains(device.name))
.toList();
final ignoredDevices = core.settings.getIgnoredDevices();
final ignoredDeviceIds = ignoredDevices.map((d) => d.id).toSet();
final newDevices = dev.where((device) {
if (devices.contains(device)) return false;
// Check if device is in the ignored list
if (device is BluetoothDevice) {
if (ignoredDeviceIds.contains(device.device.deviceId)) {
return false;
}
}
return true;
}).toList();
devices.addAll(newDevices);
_connectionQueue.addAll(newDevices);
_handleConnectionQueue();
hasDevices.value = devices.isNotEmpty;
if (devices.isNotEmpty && !_androidNotificationsSetup && !kIsWeb && Platform.isAndroid) {
_androidNotificationsSetup = true;
NotificationRequirement.setup().catchError((e) {
_actionStreams.add(LogNotification(e.toString()));
});
}
void toggleGyroscopeSteering(bool enable) {
final existing = gyroscopeDevices.firstOrNull;
if (existing != null && !enable) {
// Remove gyroscope steering
disconnect(existing, forget: true, persistForget: false);
} else if (enable) {
// Add gyroscope steering
final gyroDevice = GyroscopeSteering();
addDevices([gyroDevice]);
}
}
void _handleConnectionQueue() {
// windows apparently has issues when connecting to multiple devices at once, so don't
if (_connectionQueue.isNotEmpty && !_handlingConnectionQueue) {
if (_connectionQueue.isNotEmpty && !_handlingConnectionQueue && !screenshotMode) {
_handlingConnectionQueue = true;
final device = _connectionQueue.removeAt(0);
_actionStreams.add(LogNotification('Connecting to: ${device.name}'));
_actionStreams.add(AlertNotification(LogLevel.LOGLEVEL_INFO, 'Connecting to: ${device.name}'));
_connect(device)
.then((_) {
_handlingConnectionQueue = false;
_actionStreams.add(LogNotification('Connection finished: ${device.name}'));
_actionStreams.add(AlertNotification(LogLevel.LOGLEVEL_INFO, 'Connection finished: ${device.name}'));
if (_connectionQueue.isNotEmpty) {
_handleConnectionQueue();
}
})
.catchError((e) {
device.isConnected = false;
_handlingConnectionQueue = false;
_actionStreams.add(
LogNotification('Connection failed: ${device.name} - $e'),
);
if (e is TimeoutException) {
_actionStreams.add(
AlertNotification(LogLevel.LOGLEVEL_WARNING, 'Unable to connect to ${device.name}: Timeout'),
);
} else {
_actionStreams.add(
AlertNotification(LogLevel.LOGLEVEL_ERROR, 'Connection failed: ${device.name} - $e'),
);
}
if (_connectionQueue.isNotEmpty) {
_handleConnectionQueue();
}
@@ -254,11 +308,20 @@ class Connection {
_actionStreams.add(data);
});
if (device is BluetoothDevice) {
final connectionStateSubscription = UniversalBle.connectionStream(device.device.deviceId).listen((state) {
final connectionStateSubscription = device.device.connectionStream.listen((state) {
device.isConnected = state;
_connectionStreams.add(device);
core.flutterLocalNotificationsPlugin.show(
1338,
'${device.name} ${state ? AppLocalizations.current.connected.decapitalize() : AppLocalizations.current.disconnected.decapitalize()}',
!state ? AppLocalizations.current.tryingToConnectAgain : null,
NotificationDetails(
android: AndroidNotificationDetails('Connection', 'Connection Status'),
iOS: DarwinNotificationDetails(presentAlert: true),
),
);
if (!device.isConnected) {
disconnect(device, forget: true);
disconnect(device, forget: false, persistForget: false);
// try reconnect
performScanning();
}
@@ -269,20 +332,7 @@ class Connection {
await device.connect();
signalChange(device);
final newButtons = device.availableButtons.filter(
(button) => actionHandler.supportedApp?.keymap.getKeyPair(button) == null,
);
for (final button in newButtons) {
actionHandler.supportedApp?.keymap.addKeyPair(
KeyPair(
touchPosition: Offset.zero,
buttons: [button],
physicalKey: null,
logicalKey: null,
isLongPress: false,
),
);
}
core.actionHandler.supportedApp?.keymap.addNewButtons(device.availableButtons);
_streamSubscriptions[device] = actionSubscription;
} catch (e, backtrace) {
@@ -297,7 +347,7 @@ class Connection {
Future<void> reset() async {
_actionStreams.add(LogNotification('Disconnecting all devices'));
if (actionHandler is AndroidActions) {
if (core.actionHandler is AndroidActions) {
AndroidFlutterLocalNotificationsPlugin().stopForegroundService();
_androidNotificationsSetup = false;
}
@@ -328,39 +378,43 @@ class Connection {
_connectionStreams.add(baseDevice);
}
Future<void> disconnect(BaseDevice device, {required bool forget}) async {
Future<void> disconnect(BaseDevice device, {required bool persistForget, required bool forget}) async {
if (device.isConnected) {
await device.disconnect();
}
if (device is! LinkDevice) {
// keep it in the list to allow reconnect
devices.remove(device);
if (forget) {
_dontAllowReconnectDevices.add(device.name);
if (device is BluetoothDevice) {
if (persistForget) {
// Add device to ignored list when forgetting
await core.settings.addIgnoredDevice(device.device.deviceId, device.name);
_actionStreams.add(LogNotification('Device ignored: ${device.name}'));
}
}
if (!forget && device is BluetoothDevice) {
_lastScanResult.removeWhere((b) => b.deviceId == device.device.deviceId);
if (!forget) {
// allow reconnection
_lastScanResult.removeWhere((d) => d.deviceId == device.device.deviceId);
}
// Clean up subscriptions and scan results for reconnection
_streamSubscriptions[device]?.cancel();
_streamSubscriptions.remove(device);
_connectionSubscriptions[device]?.cancel();
_connectionSubscriptions.remove(device);
// Remove device from the list
devices.remove(device);
hasDevices.value = devices.isNotEmpty;
} else if (device is GyroscopeSteering) {
// Clean up subscriptions
_streamSubscriptions[device]?.cancel();
_streamSubscriptions.remove(device);
_connectionSubscriptions[device]?.cancel();
_connectionSubscriptions.remove(device);
// Remove device from the list
devices.remove(device);
hasDevices.value = devices.isNotEmpty;
}
signalChange(device);
}
void _onMediaKeyDetectedListener(MediaKey mediaKey) {
final hidDevice = HidDevice('HID Device');
final keyPressed = mediaKey.name;
final button = actionHandler.supportedApp!.keymap.getOrAddButton(keyPressed, () => ControllerButton(keyPressed));
var availableDevice = connection.controllerDevices.firstOrNullWhere((e) => e.name == hidDevice.name);
if (availableDevice == null) {
connection.addDevices([hidDevice]);
availableDevice = hidDevice;
}
availableDevice.handleButtonsClicked([button]);
availableDevice.handleButtonsClicked([]);
}
}

View File

@@ -1,9 +1,16 @@
import 'dart:async';
import 'package:bike_control/bluetooth/devices/zwift/constants.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zp.pb.dart' show LogLevel;
import 'package:bike_control/gen/l10n.dart';
import 'package:bike_control/utils/actions/desktop.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/iap/iap_manager.dart';
import 'package:bike_control/utils/keymap/apps/custom_app.dart';
import 'package:bike_control/utils/keymap/manager.dart';
import 'package:dartx/dartx.dart';
import 'package:flutter/material.dart';
import 'package:swift_control/bluetooth/devices/zwift/constants.dart';
import 'package:swift_control/main.dart';
import 'package:swift_control/utils/actions/desktop.dart';
import 'package:flutter_local_notifications/flutter_local_notifications.dart';
import '../../utils/keymap/buttons.dart';
import '../messages/notification.dart';
@@ -13,7 +20,13 @@ abstract class BaseDevice {
final bool isBeta;
final List<ControllerButton> availableButtons;
BaseDevice(this.name, {required this.availableButtons, this.isBeta = false});
BaseDevice(this.name, {required this.availableButtons, this.isBeta = false}) {
if (availableButtons.isEmpty && core.actionHandler.supportedApp is CustomApp) {
// TODO we should verify where the buttons came from
final allButtons = core.actionHandler.supportedApp!.keymap.keyPairs.flatMap((e) => e.buttons);
availableButtons.addAll(allButtons);
}
}
bool isConnected = false;
@@ -38,7 +51,34 @@ abstract class BaseDevice {
Future<void> connect();
Future<void> handleButtonsClicked(List<ControllerButton>? buttonsClicked) async {
Future<void> handleButtonsClickedWithoutLongPressSupport(List<ControllerButton> clickedButtons) async {
await handleButtonsClicked(clickedButtons, longPress: true);
if (clickedButtons.length == 1) {
final keyPair = core.actionHandler.supportedApp?.keymap.getKeyPair(clickedButtons.single);
if (keyPair != null && (keyPair.isLongPress || keyPair.inGameAction?.isLongPress == true)) {
// simulate release after click
_longPressTimer?.cancel();
await Future.delayed(const Duration(milliseconds: 800));
await handleButtonsClicked([], longPress: true);
} else {
await handleButtonsClicked([], longPress: true);
}
} else {
await handleButtonsClicked([]);
}
}
Future<void> handleButtonsClicked(List<ControllerButton>? buttonsClicked, {bool longPress = false}) async {
try {
await _handleButtonsClickedInternal(buttonsClicked, longPress: longPress);
} catch (e, st) {
actionStreamInternal.add(
LogNotification('Error handling button clicks: $e\n$st'),
);
}
}
Future<void> _handleButtonsClickedInternal(List<ControllerButton>? buttonsClicked, {required bool longPress}) async {
if (buttonsClicked == null) {
// ignore, no changes
} else if (buttonsClicked.isEmpty) {
@@ -48,8 +88,9 @@ abstract class BaseDevice {
// Handle release events for long press keys
final buttonsReleased = _previouslyPressedButtons.toList();
final isLongPress =
longPress ||
buttonsReleased.singleOrNull != null &&
actionHandler.supportedApp?.keymap.getKeyPair(buttonsReleased.single)?.isLongPress == true;
core.actionHandler.supportedApp?.keymap.getKeyPair(buttonsReleased.single)?.isLongPress == true;
if (buttonsReleased.isNotEmpty && isLongPress) {
await performRelease(buttonsReleased);
}
@@ -60,15 +101,17 @@ abstract class BaseDevice {
// Handle release events for buttons that are no longer pressed
final buttonsReleased = _previouslyPressedButtons.difference(buttonsClicked.toSet()).toList();
final wasLongPress =
longPress ||
buttonsReleased.singleOrNull != null &&
actionHandler.supportedApp?.keymap.getKeyPair(buttonsReleased.single)?.isLongPress == true;
core.actionHandler.supportedApp?.keymap.getKeyPair(buttonsReleased.single)?.isLongPress == true;
if (buttonsReleased.isNotEmpty && wasLongPress) {
await performRelease(buttonsReleased);
}
final isLongPress =
longPress ||
buttonsClicked.singleOrNull != null &&
actionHandler.supportedApp?.keymap.getKeyPair(buttonsClicked.single)?.isLongPress == true;
core.actionHandler.supportedApp?.keymap.getKeyPair(buttonsClicked.single)?.isLongPress == true;
if (!isLongPress &&
!(buttonsClicked.singleOrNull == ZwiftButtons.onOffLeft ||
@@ -90,40 +133,113 @@ abstract class BaseDevice {
}
}
String _getCommandLimitMessage() {
return AppLocalizations.current.dailyCommandLimitReachedNotification;
}
String _getCommandLimitTitle() {
return AppLocalizations.current
.dailyLimitReached(IAPManager.dailyCommandLimit, IAPManager.dailyCommandLimit)
.replaceAll(
'${IAPManager.dailyCommandLimit}/${IAPManager.dailyCommandLimit}',
IAPManager.dailyCommandLimit.toString(),
)
.replaceAll(
'${IAPManager.dailyCommandLimit} / ${IAPManager.dailyCommandLimit}',
IAPManager.dailyCommandLimit.toString(),
);
}
Future<void> performDown(List<ControllerButton> buttonsClicked) async {
for (final action in buttonsClicked) {
// Check IAP status before executing command
if (!IAPManager.instance.canExecuteCommand) {
//actionStreamInternal.add(AlertNotification(LogLevel.LOGLEVEL_ERROR, _getCommandLimitMessage()));
continue;
}
// For repeated actions, don't trigger key down/up events (useful for long press)
actionStreamInternal.add(
LogNotification(await actionHandler.performAction(action, isKeyDown: true, isKeyUp: false)),
);
final result = await core.actionHandler.performAction(action, isKeyDown: true, isKeyUp: false);
actionStreamInternal.add(ActionNotification(result));
}
}
Future<void> performClick(List<ControllerButton> buttonsClicked) async {
for (final action in buttonsClicked) {
actionStreamInternal.add(
LogNotification(await actionHandler.performAction(action, isKeyDown: true, isKeyUp: true)),
);
// Check IAP status before executing command
if (!IAPManager.instance.canExecuteCommand) {
_showCommandLimitAlert();
continue;
}
final result = await core.actionHandler.performAction(action, isKeyDown: true, isKeyUp: true);
actionStreamInternal.add(ActionNotification(result));
}
}
Future<void> performRelease(List<ControllerButton> buttonsReleased) async {
for (final action in buttonsReleased) {
actionStreamInternal.add(
LogNotification(await actionHandler.performAction(action, isKeyDown: false, isKeyUp: true)),
);
// Check IAP status before executing command
if (!IAPManager.instance.canExecuteCommand) {
_showCommandLimitAlert();
continue;
}
final result = await core.actionHandler.performAction(action, isKeyDown: false, isKeyUp: true);
actionStreamInternal.add(LogNotification(result.message));
}
}
Future<void> disconnect() async {
_longPressTimer?.cancel();
// Release any held keys in long press mode
if (actionHandler is DesktopActions) {
await (actionHandler as DesktopActions).releaseAllHeldKeys(_previouslyPressedButtons.toList());
if (core.actionHandler is DesktopActions) {
await (core.actionHandler as DesktopActions).releaseAllHeldKeys(_previouslyPressedButtons.toList());
}
_previouslyPressedButtons.clear();
isConnected = false;
}
Widget showInformation(BuildContext context);
ControllerButton getOrAddButton(String key, ControllerButton Function() creator) {
if (core.actionHandler.supportedApp == null) {
return creator();
}
if (core.actionHandler.supportedApp is! CustomApp) {
final currentProfile = core.actionHandler.supportedApp!.name;
// should we display this to the user?
KeymapManager().duplicateSync(currentProfile, '$currentProfile (Copy)');
}
final button = core.actionHandler.supportedApp!.keymap.getOrAddButton(key, creator);
if (availableButtons.none((e) => e.name == button.name)) {
availableButtons.add(button);
core.settings.setKeyMap(core.actionHandler.supportedApp!);
}
return button;
}
void _showCommandLimitAlert() {
actionStreamInternal.add(
AlertNotification(
LogLevel.LOGLEVEL_ERROR,
_getCommandLimitMessage(),
buttonTitle: AppLocalizations.current.purchase,
onTap: () {
IAPManager.instance.purchaseFullVersion();
},
),
);
core.flutterLocalNotificationsPlugin.show(
1337,
_getCommandLimitTitle(),
_getCommandLimitMessage(),
NotificationDetails(
android: AndroidNotificationDetails('Limit', 'Limit reached'),
iOS: DarwinNotificationDetails(presentAlert: true),
),
);
}
}

View File

@@ -1,21 +1,29 @@
import 'dart:async';
import 'package:bike_control/bluetooth/ble.dart';
import 'package:bike_control/bluetooth/devices/base_device.dart';
import 'package:bike_control/bluetooth/devices/openbikecontrol/openbikecontrol_device.dart';
import 'package:bike_control/bluetooth/devices/shimano/shimano_di2.dart';
import 'package:bike_control/bluetooth/devices/sram/sram_axs.dart';
import 'package:bike_control/bluetooth/devices/wahoo/wahoo_kickr_bike_pro.dart';
import 'package:bike_control/bluetooth/devices/wahoo/wahoo_kickr_bike_shift.dart';
import 'package:bike_control/bluetooth/devices/wahoo/wahoo_kickr_headwind.dart';
import 'package:bike_control/bluetooth/devices/zwift/constants.dart';
import 'package:bike_control/bluetooth/devices/zwift/zwift_click.dart';
import 'package:bike_control/bluetooth/devices/zwift/zwift_clickv2.dart';
import 'package:bike_control/bluetooth/devices/zwift/zwift_device.dart';
import 'package:bike_control/bluetooth/devices/zwift/zwift_play.dart';
import 'package:bike_control/bluetooth/devices/zwift/zwift_ride.dart';
import 'package:bike_control/pages/device.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/i18n_extension.dart';
import 'package:bike_control/widgets/ui/beta_pill.dart';
import 'package:bike_control/widgets/ui/device_info.dart';
import 'package:bike_control/widgets/ui/loading_widget.dart';
import 'package:bike_control/widgets/ui/small_progress_indicator.dart';
import 'package:dartx/dartx.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:swift_control/bluetooth/ble.dart';
import 'package:swift_control/bluetooth/devices/base_device.dart';
import 'package:swift_control/bluetooth/devices/shimano/shimano_di2.dart';
import 'package:swift_control/bluetooth/devices/wahoo/wahoo_kickr_bike_shift.dart';
import 'package:swift_control/bluetooth/devices/zwift/constants.dart';
import 'package:swift_control/bluetooth/devices/zwift/zwift_click.dart';
import 'package:swift_control/bluetooth/devices/zwift/zwift_clickv2.dart';
import 'package:swift_control/bluetooth/devices/zwift/zwift_device.dart';
import 'package:swift_control/bluetooth/devices/zwift/zwift_play.dart';
import 'package:swift_control/bluetooth/devices/zwift/zwift_ride.dart';
import 'package:swift_control/main.dart';
import 'package:swift_control/pages/device.dart';
import 'package:swift_control/widgets/beta_pill.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
import 'package:universal_ble/universal_ble.dart';
import 'cycplus/cycplus_bc2.dart';
@@ -39,9 +47,12 @@ abstract class BluetoothDevice extends BaseDevice {
ZwiftConstants.ZWIFT_RIDE_CUSTOM_SERVICE_UUID,
SquareConstants.SERVICE_UUID,
WahooKickrBikeShiftConstants.SERVICE_UUID,
WahooKickrHeadwindConstants.SERVICE_UUID,
SterzoConstants.SERVICE_UUID,
CycplusBc2Constants.SERVICE_UUID,
ShimanoDi2Constants.SERVICE_UUID,
ShimanoDi2Constants.SERVICE_UUID_ALTERNATIVE,
OpenBikeControlConstants.SERVICE_UUID,
];
static BluetoothDevice? fromScanResult(BleDevice scanResult) {
@@ -53,12 +64,16 @@ abstract class BluetoothDevice extends BaseDevice {
'Zwift Play' => ZwiftPlay(scanResult),
'Zwift Click' => ZwiftClickV2(scanResult),
'SQUARE' => EliteSquare(scanResult),
'OpenBike' => OpenBikeControlDevice(scanResult),
null => null,
_ when scanResult.name!.toUpperCase().startsWith('HEADWIND') => WahooKickrHeadwind(scanResult),
_ when scanResult.name!.toUpperCase().startsWith('STERZO') => EliteSterzo(scanResult),
_ when scanResult.name!.toUpperCase().startsWith('KICKR BIKE SHIFT') => WahooKickrBikeShift(scanResult),
_ when scanResult.name!.toUpperCase().startsWith('CYCPLUS') || scanResult.name!.toUpperCase().contains('BC2') =>
_ when scanResult.name!.toUpperCase().startsWith('KICKR BIKE PRO') => WahooKickrBikePro(scanResult),
_ when scanResult.name!.toUpperCase().startsWith('CYCPLUS') && scanResult.name!.toUpperCase().contains('BC2') =>
CycplusBc2(scanResult),
_ when scanResult.name!.toUpperCase().startsWith('RDR') => ShimanoDi2(scanResult),
_ when scanResult.name!.toUpperCase().startsWith('SRAM') => SramAxs(scanResult),
_ => null,
};
} else {
@@ -68,13 +83,25 @@ abstract class BluetoothDevice extends BaseDevice {
// https://www.makinolo.com/blog/2024/07/26/zwift-ride-protocol/
'Zwift Play' => ZwiftPlay(scanResult),
//'Zwift Click' => ZwiftClick(scanResult), special case for Zwift Click v2: we must only connect to the left controller
_ when scanResult.name!.toUpperCase().startsWith('HEADWIND') => WahooKickrHeadwind(scanResult),
_ when scanResult.name!.toUpperCase().startsWith('SQUARE') => EliteSquare(scanResult),
_ when scanResult.name!.toUpperCase().startsWith('STERZO') => EliteSterzo(scanResult),
_ when scanResult.name!.toUpperCase().contains('KICKR BIKE SHIFT') => WahooKickrBikeShift(scanResult),
_ when scanResult.name!.toUpperCase().startsWith('CYCPLUS') || scanResult.name!.toUpperCase().contains('BC2') =>
_ when scanResult.name!.toUpperCase().startsWith('KICKR BIKE PRO') => WahooKickrBikePro(scanResult),
_ when scanResult.name!.toUpperCase().startsWith('CYCPLUS') && scanResult.name!.toUpperCase().contains('BC2') =>
CycplusBc2(scanResult),
_ when scanResult.services.contains(CycplusBc2Constants.SERVICE_UUID.toLowerCase()) => CycplusBc2(scanResult),
_ when scanResult.services.contains(ShimanoDi2Constants.SERVICE_UUID.toLowerCase()) => ShimanoDi2(scanResult),
_ when scanResult.services.contains(ShimanoDi2Constants.SERVICE_UUID_ALTERNATIVE.toLowerCase()) => ShimanoDi2(
scanResult,
),
_ when scanResult.services.contains(SramAxsConstants.SERVICE_UUID.toLowerCase()) => SramAxs(
scanResult,
),
_ when scanResult.services.contains(OpenBikeControlConstants.SERVICE_UUID.toLowerCase()) =>
OpenBikeControlDevice(scanResult),
_ when scanResult.services.contains(WahooKickrHeadwindConstants.SERVICE_UUID.toLowerCase()) =>
WahooKickrHeadwind(scanResult),
// otherwise the service UUIDs will be used
_ => null,
};
@@ -105,6 +132,11 @@ abstract class BluetoothDevice extends BaseDevice {
//DeviceType.rideRight => ZwiftRide(scanResult), // see comment above
ZwiftDeviceType.clickV2Left => ZwiftClickV2(scanResult),
//DeviceType.clickV2Right => ZwiftClickV2(scanResult), // see comment above
_
when scanResult.name == 'Zwift Ride' &&
type != ZwiftDeviceType.rideRight &&
type != ZwiftDeviceType.rideLeft =>
ZwiftRide(scanResult), // e.g. old firmware
_ => null,
};
} else {
@@ -115,10 +147,10 @@ abstract class BluetoothDevice extends BaseDevice {
@override
bool operator ==(Object other) =>
identical(this, other) ||
other is BluetoothDevice && runtimeType == other.runtimeType && scanResult == other.scanResult;
other is BluetoothDevice && runtimeType == other.runtimeType && scanResult.deviceId == other.scanResult.deviceId;
@override
int get hashCode => scanResult.hashCode;
int get hashCode => scanResult.deviceId.hashCode;
@override
String toString() {
@@ -129,11 +161,12 @@ abstract class BluetoothDevice extends BaseDevice {
@override
Future<void> connect() async {
actionStream.listen((message) {
print("Received message: $message");
});
await UniversalBle.connect(device.deviceId);
try {
await UniversalBle.connect(device.deviceId);
} catch (e) {
isConnected = false;
rethrow;
}
if (!kIsWeb) {
await UniversalBle.requestMtu(device.deviceId, 517);
@@ -153,7 +186,8 @@ abstract class BluetoothDevice extends BaseDevice {
firmwareCharacteristic.uuid,
);
firmwareVersion = String.fromCharCodes(firmwareData);
connection.signalChange(this);
core.connection.signalChange(this);
}
final batteryService = services.firstOrNullWhere(
@@ -171,7 +205,7 @@ abstract class BluetoothDevice extends BaseDevice {
);
if (batteryData.isNotEmpty) {
batteryLevel = batteryData.first;
connection.signalChange(this);
core.connection.signalChange(this);
}
}
@@ -189,63 +223,127 @@ abstract class BluetoothDevice extends BaseDevice {
@override
Widget showInformation(BuildContext context) {
return Row(
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
device.name?.screenshot ?? device.runtimeType.toString(),
style: TextStyle(fontWeight: FontWeight.bold),
Row(
spacing: 8,
children: [
Text(
device.name?.screenshot ?? runtimeType.toString(),
style: TextStyle(fontWeight: FontWeight.bold),
),
if (isBeta) BetaPill(),
Expanded(child: SizedBox()),
Builder(
builder: (context) {
return LoadingWidget(
futureCallback: () async {
final completer = showDropdown<bool>(
context: context,
builder: (c) => DropdownMenu(
children: [
MenuButton(
child: Text('Disconnect and Forget for this session'),
onPressed: (context) {
closeOverlay(context, false);
},
),
MenuButton(
child: Text('Disconnect and Forget'),
onPressed: (context) {
closeOverlay(context, true);
},
),
],
),
);
final persist = await completer.future;
if (persist != null) {
await core.connection.disconnect(this, forget: true, persistForget: persist);
}
},
renderChild: (isLoading, tap) => IconButton(
variance: ButtonVariance.muted,
icon: isLoading ? SmallProgressIndicator() : Icon(Icons.clear),
onPressed: tap,
),
);
},
),
],
),
if (isBeta) BetaPill(),
if (batteryLevel != null) ...[
Icon(switch (batteryLevel!) {
>= 80 => Icons.battery_full,
>= 60 => Icons.battery_6_bar,
>= 50 => Icons.battery_5_bar,
>= 25 => Icons.battery_4_bar,
>= 10 => Icons.battery_2_bar,
_ => Icons.battery_alert,
}),
Text('$batteryLevel%'),
],
if (firmwareVersion != null) Text(' - v$firmwareVersion'),
if (firmwareVersion != null &&
this is ZwiftDevice &&
firmwareVersion != (this as ZwiftDevice).latestFirmwareVersion) ...[
SizedBox(width: 8),
Icon(Icons.warning, color: Theme.of(context).colorScheme.error),
Text(
' (latest: ${(this as ZwiftDevice).latestFirmwareVersion})',
style: TextStyle(color: Theme.of(context).colorScheme.error),
),
],
if (rssi != null)
Padding(
padding: const EdgeInsets.only(left: 8.0),
child: Tooltip(
message: 'Signal Strength: $rssi dBm',
child: Icon(
switch (rssi!) {
Wrap(
spacing: 12,
runSpacing: 12,
children: [
DeviceInfo(
title: context.i18n.connection,
icon: switch (isConnected) {
true => Icons.bluetooth_connected_outlined,
false => Icons.bluetooth_disabled_outlined,
},
value: isConnected ? context.i18n.connected : context.i18n.disconnected,
),
if (batteryLevel != null)
DeviceInfo(
title: context.i18n.battery,
icon: switch (batteryLevel!) {
>= 80 => Icons.battery_full,
>= 60 => Icons.battery_6_bar,
>= 50 => Icons.battery_5_bar,
>= 25 => Icons.battery_4_bar,
>= 10 => Icons.battery_2_bar,
_ => Icons.battery_alert,
},
value: '$batteryLevel%',
),
if (firmwareVersion != null)
DeviceInfo(
title: context.i18n.signal,
icon: this is ZwiftDevice && firmwareVersion != (this as ZwiftDevice).latestFirmwareVersion
? Icons.warning
: Icons.text_fields_sharp,
value: firmwareVersion!,
additionalInfo: (this is ZwiftDevice && firmwareVersion != (this as ZwiftDevice).latestFirmwareVersion)
? Text(
' (${context.i18n.latestVersion((this as ZwiftDevice).latestFirmwareVersion)})',
style: TextStyle(color: Theme.of(context).colorScheme.destructive, fontSize: 12),
)
: null,
),
if (rssi != null)
DeviceInfo(
title: context.i18n.signal,
icon: switch (rssi!) {
>= -50 => Icons.signal_cellular_4_bar,
>= -60 => Icons.signal_cellular_alt_2_bar,
>= -70 => Icons.signal_cellular_alt_1_bar,
_ => Icons.signal_cellular_alt,
},
size: 18,
value: '$rssi dBm',
),
),
),
Expanded(child: SizedBox()),
PopupMenuButton(
itemBuilder: (c) => [
PopupMenuItem(
child: Text('Disconnect and Forget'),
onTap: () {
connection.disconnect(this, forget: true);
},
),
],
),
],
);
}
void debugSubscribeToAll(List<BleService> services) {
for (final service in services) {
for (final characteristic in service.characteristics) {
if (characteristic.properties.contains(CharacteristicProperty.indicate)) {
debugPrint('Subscribing to indications for ${service.uuid} / ${characteristic.uuid}');
UniversalBle.subscribeIndications(device.deviceId, service.uuid, characteristic.uuid);
}
if (characteristic.properties.contains(CharacteristicProperty.notify)) {
debugPrint('Subscribing to notifications for ${service.uuid} / ${characteristic.uuid}');
UniversalBle.subscribeNotifications(device.deviceId, service.uuid, characteristic.uuid);
}
}
}
}
}

View File

@@ -1,8 +1,8 @@
import 'dart:typed_data';
import 'package:flutter/material.dart';
import 'package:swift_control/bluetooth/messages/notification.dart';
import 'package:swift_control/utils/keymap/buttons.dart';
import 'package:bike_control/bluetooth/messages/notification.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:universal_ble/universal_ble.dart';
import '../bluetooth_device.dart';
@@ -27,43 +27,74 @@ class CycplusBc2 extends BluetoothDevice {
await UniversalBle.subscribeNotifications(device.deviceId, service.uuid, characteristic.uuid);
}
// Track last state for index 6 and 7
int _lastStateIndex6 = 0x00;
int _lastStateIndex7 = 0x00;
@override
Future<void> processCharacteristic(String characteristic, Uint8List bytes) {
if (characteristic.toLowerCase() == CycplusBc2Constants.TX_CHARACTERISTIC_UUID.toLowerCase()) {
// Process CYCPLUS BC2 data
// The BC2 typically sends button press data as simple byte values
// Common patterns for virtual shifters:
// - 0x01 or similar for shift up
// - 0x02 or similar for shift down
// - 0x00 for button release
if (bytes.length > 7) {
final buttonsToPress = <ControllerButton>[];
if (bytes.isNotEmpty) {
final buttonCode = bytes[0];
switch (buttonCode) {
case 0x01:
// Shift up button pressed
handleButtonsClicked([CycplusBc2Buttons.shiftUp]);
break;
case 0x02:
// Shift down button pressed
handleButtonsClicked([CycplusBc2Buttons.shiftDown]);
break;
case 0x00:
// Button released
handleButtonsClicked([]);
break;
default:
// Unknown button code - log for debugging
actionStreamInternal.add(
LogNotification('CYCPLUS BC2: Unknown button code: 0x${buttonCode.toRadixString(16)}'),
);
break;
// Process index 6 (shift up)
final currentByte6 = bytes[6];
if (_shouldTriggerShift(currentByte6, _lastStateIndex6)) {
buttonsToPress.add(CycplusBc2Buttons.shiftUp);
_lastStateIndex6 = 0x00; // Reset after successful press
} else {
_updateState(currentByte6, (val) => _lastStateIndex6 = val);
}
// Process index 7 (shift down)
final currentByte7 = bytes[7];
if (_shouldTriggerShift(currentByte7, _lastStateIndex7)) {
buttonsToPress.add(CycplusBc2Buttons.shiftDown);
_lastStateIndex7 = 0x00; // Reset after successful press
} else {
_updateState(currentByte7, (val) => _lastStateIndex7 = val);
}
handleButtonsClicked(buttonsToPress);
} else {
actionStreamInternal.add(
LogNotification(
'CYCPLUS BC2 received unexpected packet: ${bytes.map((e) => e.toRadixString(16).padLeft(2, '0')).join()}',
),
);
handleButtonsClicked([]);
}
}
return Future.value();
}
// Check if we should trigger a shift based on current and last state
bool _shouldTriggerShift(int currentByte, int lastByte) {
const pressedValues = {0x01, 0x02, 0x03};
// State change from one pressed value to another different pressed value
// This is the ONLY time we trigger a shift
if (pressedValues.contains(currentByte) && pressedValues.contains(lastByte) && currentByte != lastByte) {
return true;
}
return false;
}
// Update state tracking
void _updateState(int currentByte, void Function(int) setState) {
const pressedValues = {0x01, 0x02, 0x03};
const releaseValue = 0x00;
// Button released: current is 0x00 and last was pressed
if (currentByte == releaseValue) {
setState(releaseValue);
}
// Lock the new pressed state
else if (pressedValues.contains(currentByte)) {
setState(currentByte);
}
}
}
class CycplusBc2Constants {

View File

@@ -1,7 +1,6 @@
import 'dart:typed_data';
import 'package:dartx/dartx.dart';
import 'package:swift_control/utils/keymap/buttons.dart';
import 'package:flutter/foundation.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:universal_ble/universal_ble.dart';
import '../../messages/notification.dart';
@@ -37,19 +36,19 @@ class EliteSquare extends BluetoothDevice {
if (characteristic == SquareConstants.CHARACTERISTIC_UUID) {
final fullValue = _bytesToHex(bytes);
final currentValue = _extractButtonCode(fullValue);
actionStreamInternal.add(LogNotification('Received $fullValue - vs $currentValue (last: $_lastValue)'));
if (kDebugMode) {
actionStreamInternal.add(LogNotification('Received $fullValue - vs $currentValue (last: $_lastValue)'));
}
if (_lastValue != null) {
final currentRelevantPart = fullValue.length >= 14
? fullValue.substring(6, 14)
: fullValue.substring(6);
final lastRelevantPart = _lastValue!.length >= 14
? _lastValue!.substring(6, 14)
: _lastValue!.substring(6);
final currentRelevantPart = fullValue.length >= 14 ? fullValue.substring(6, 14) : fullValue.substring(6);
final lastRelevantPart = _lastValue!.length >= 14 ? _lastValue!.substring(6, 14) : _lastValue!.substring(6);
if (currentRelevantPart != lastRelevantPart) {
final buttonClicked = SquareConstants.BUTTON_MAPPING[currentValue];
actionStreamInternal.add(LogNotification('Button pressed: $buttonClicked'));
if (kDebugMode) {
actionStreamInternal.add(LogNotification('Button pressed: $buttonClicked'));
}
handleButtonsClicked([
if (buttonClicked != null) buttonClicked,
]);

View File

@@ -5,8 +5,8 @@ import 'package:dartx/dartx.dart';
import 'package:flutter/foundation.dart';
import 'package:http/http.dart' as http;
import 'package:shared_preferences/shared_preferences.dart';
import 'package:swift_control/bluetooth/devices/bluetooth_device.dart';
import 'package:swift_control/utils/keymap/buttons.dart';
import 'package:bike_control/bluetooth/devices/bluetooth_device.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:universal_ble/universal_ble.dart';
import '../../messages/notification.dart';

View File

@@ -1,38 +1,59 @@
import 'dart:io';
import 'package:dartx/dartx.dart';
import 'package:flutter/material.dart';
import 'package:gamepads/gamepads.dart';
import 'package:swift_control/bluetooth/devices/base_device.dart';
import 'package:swift_control/bluetooth/messages/notification.dart';
import 'package:swift_control/main.dart';
import 'package:swift_control/pages/device.dart';
import 'package:swift_control/utils/keymap/apps/custom_app.dart';
import 'package:swift_control/utils/keymap/buttons.dart';
import 'package:swift_control/widgets/beta_pill.dart';
import '../../../widgets/warning.dart';
import 'package:bike_control/bluetooth/devices/base_device.dart';
import 'package:bike_control/bluetooth/messages/notification.dart';
import 'package:bike_control/pages/device.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:bike_control/widgets/ui/beta_pill.dart';
class GamepadDevice extends BaseDevice {
final String id;
GamepadDevice(super.name, {required this.id}) : super(availableButtons: [], isBeta: true);
GamepadDevice(super.name, {required this.id}) : super(availableButtons: []);
List<ControllerButton> _lastButtonsClicked = [];
@override
Future<void> connect() async {
Gamepads.eventsByGamepad(id).listen((event) {
actionStreamInternal.add(LogNotification('Gamepad event: $event'));
Gamepads.eventsByGamepad(id).listen((event) async {
actionStreamInternal.add(LogNotification('Gamepad event: ${event.key} value ${event.value} type ${event.type}'));
ControllerButton? button = actionHandler.supportedApp?.keymap.getOrAddButton(
event.key,
() => ControllerButton(event.key),
final int normalizedValue = switch (event.value) {
> 1.0 => 1,
< -1.0 => -1,
_ => event.value.toInt(),
};
final buttonKey = event.type == KeyType.analog ? '${event.key}_$normalizedValue' : event.key;
ControllerButton button = getOrAddButton(
buttonKey,
() => ControllerButton(buttonKey),
);
final buttonsClicked = event.value == 0.0 && button != null ? [button] : <ControllerButton>[];
if (_lastButtonsClicked.contentEquals(buttonsClicked) == false) {
handleButtonsClicked(buttonsClicked);
switch (event.type) {
case KeyType.analog:
final releasedValue = Platform.isWindows ? 1 : 0;
if (event.value.round().abs() != releasedValue) {
final buttonsClicked = [button];
if (_lastButtonsClicked.contentEquals(buttonsClicked) == false) {
handleButtonsClicked(buttonsClicked);
}
_lastButtonsClicked = buttonsClicked;
} else {
_lastButtonsClicked = [];
handleButtonsClicked([]);
}
case KeyType.button:
final buttonsClicked = event.value.toInt() != 1 ? [button] : <ControllerButton>[];
if (_lastButtonsClicked.contentEquals(buttonsClicked) == false) {
handleButtonsClicked(buttonsClicked);
}
_lastButtonsClicked = buttonsClicked;
}
_lastButtonsClicked = buttonsClicked;
});
}
@@ -44,6 +65,7 @@ class GamepadDevice extends BaseDevice {
spacing: 8,
children: [
Row(
spacing: 8,
children: [
Text(
name.screenshot,
@@ -52,13 +74,6 @@ class GamepadDevice extends BaseDevice {
if (isBeta) BetaPill(),
],
),
if (actionHandler.supportedApp is! CustomApp)
Warning(
children: [
Text('Use a custom keymap to use the buttons on $name.'),
],
),
],
),
);

View File

@@ -0,0 +1,315 @@
import 'dart:async';
import 'package:bike_control/bluetooth/devices/base_device.dart';
import 'package:bike_control/bluetooth/devices/gyroscope/steering_estimator.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zp.pb.dart';
import 'package:bike_control/bluetooth/messages/notification.dart';
import 'package:bike_control/pages/device.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:bike_control/widgets/ui/beta_pill.dart';
import 'package:bike_control/widgets/ui/device_info.dart';
import 'package:bike_control/widgets/ui/small_progress_indicator.dart';
import 'package:flutter/foundation.dart';
import 'package:sensors_plus/sensors_plus.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
/// Gyroscope and Accelerometer based steering device
/// Detects handlebar movement when the phone is mounted on the handlebar
class GyroscopeSteering extends BaseDevice {
GyroscopeSteering()
: super(
'Phone Steering',
availableButtons: GyroscopeSteeringButtons.values,
isBeta: true,
);
StreamSubscription<GyroscopeEvent>? _gyroscopeSubscription;
StreamSubscription<AccelerometerEvent>? _accelerometerSubscription;
// Calibration state
final SteeringEstimator _estimator = SteeringEstimator();
bool _isCalibrated = false;
ControllerButton? _lastSteeringButton;
// Accelerometer raw data
bool _hasAccelData = false;
// Time tracking for integration
DateTime? _lastGyroUpdate;
// Last rounded angle for change detection
int? _lastRoundedAngle;
// Debounce timer for PWM-like keypress behavior
Timer? _keypressTimer;
bool _isProcessingKeypresses = false;
// Configuration (can be made customizable later)
static const double STEERING_THRESHOLD = 5.0; // degrees
static const double LEVEL_DEGREE_STEP = 10.0; // degrees per level
static const int MAX_LEVELS = 5;
static const int KEY_REPEAT_INTERVAL_MS = 40;
static const double COMPLEMENTARY_FILTER_ALPHA = 0.98; // Weight for gyroscope
static const double LOW_PASS_FILTER_ALPHA = 0.9; // Smoothing factor
@override
Future<void> connect() async {
if (isConnected) {
return;
}
try {
// Start listening to sensors
_gyroscopeSubscription = gyroscopeEventStream().listen(
_handleGyroscopeEvent,
onError: (error) {
actionStreamInternal.add(LogNotification('Gyroscope error: $error'));
},
);
_accelerometerSubscription = accelerometerEventStream().listen(
_handleAccelerometerEvent,
onError: (error) {
actionStreamInternal.add(LogNotification('Accelerometer error: $error'));
},
);
isConnected = true;
actionStreamInternal.add(LogNotification('Gyroscope Steering: Connected - Calibrating...'));
// Reset calibration/estimator
_isCalibrated = false;
_hasAccelData = false;
_estimator.reset();
_lastGyroUpdate = null;
_lastRoundedAngle = null;
_lastSteeringButton = null;
} catch (e) {
actionStreamInternal.add(LogNotification('Failed to connect Gyroscope Steering: $e'));
isConnected = false;
rethrow;
}
}
void _handleGyroscopeEvent(GyroscopeEvent event) {
final now = DateTime.now();
if (!_hasAccelData) {
_lastGyroUpdate = now;
return;
}
final dt = _lastGyroUpdate != null ? (now.difference(_lastGyroUpdate!).inMicroseconds / 1000000.0) : 0.0;
_lastGyroUpdate = now;
if (dt <= 0 || dt >= 1.0) {
return;
}
// iOS drift fix:
// - integrate bias-corrected gyro z (yaw) into an estimator
// - learn bias while the device is still
final angleDeg = _estimator.updateGyro(wz: event.z, dt: dt);
if (!_isCalibrated) {
// Consider calibration complete once we have a bit of stillness and sensor data.
// This gives the bias estimator time to settle.
if (_estimator.stillTimeSec >= 0.6) {
_estimator.calibrate(seedBiasZRadPerSec: _estimator.biasZRadPerSec);
_isCalibrated = true;
actionStreamInternal.add(
AlertNotification(LogLevel.LOGLEVEL_INFO, 'Calibration complete.'),
);
}
return;
}
_processSteeringAngle(angleDeg);
}
void _handleAccelerometerEvent(AccelerometerEvent event) {
_hasAccelData = true;
_estimator.updateAccel(x: event.x, y: event.y, z: event.z);
}
void _processSteeringAngle(double steeringAngleDeg) {
final roundedAngle = steeringAngleDeg.round();
if (_lastRoundedAngle != roundedAngle) {
if (kDebugMode) {
actionStreamInternal.add(
LogNotification(
'Steering angle: $roundedAngle° (biasZ=${_estimator.biasZRadPerSec.toStringAsFixed(4)} rad/s)',
),
);
}
_lastRoundedAngle = roundedAngle;
_applyPWMSteering(roundedAngle);
}
}
/// Applies PWM-like steering behavior with repeated keypresses proportional to angle magnitude
void _applyPWMSteering(int roundedAngle) {
// Cancel any pending keypress timer
_keypressTimer?.cancel();
// Determine if we're steering
if (roundedAngle.abs() > core.settings.getPhoneSteeringThreshold()) {
// Determine direction
final button = roundedAngle < 0 ? GyroscopeSteeringButtons.rightSteer : GyroscopeSteeringButtons.leftSteer;
if (_lastSteeringButton != button) {
// New steering direction - reset any previous state
_lastSteeringButton = button;
} else {
return;
}
handleButtonsClicked([button]);
} else {
_lastSteeringButton = null;
// Center position - release any held buttons
handleButtonsClicked([]);
}
}
@override
Future<void> disconnect() async {
await _gyroscopeSubscription?.cancel();
await _accelerometerSubscription?.cancel();
_gyroscopeSubscription = null;
_accelerometerSubscription = null;
_keypressTimer?.cancel();
isConnected = false;
_isCalibrated = false;
_hasAccelData = false;
_estimator.reset();
actionStreamInternal.add(LogNotification('Gyroscope Steering: Disconnected'));
}
@override
Widget showInformation(BuildContext context) {
return StatefulBuilder(
builder: (c, setState) => Column(
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
spacing: 12,
children: [
Row(
spacing: 12,
children: [
Text(
name.screenshot,
style: TextStyle(fontWeight: FontWeight.bold),
),
if (isBeta) BetaPill(),
],
),
Wrap(
spacing: 12,
runSpacing: 12,
children: [
DeviceInfo(
title: 'Calibration',
icon: BootstrapIcons.wrenchAdjustable,
value: _isCalibrated ? 'Complete' : 'In Progress',
),
DeviceInfo(
title: 'Steering Angle',
icon: RadixIcons.angle,
value: _isCalibrated ? '${_estimator.angleDeg.toStringAsFixed(2)}°' : 'Calibrating...',
),
if (kDebugMode)
DeviceInfo(
title: 'Gyro Bias',
icon: BootstrapIcons.speedometer,
value: '${_estimator.biasZRadPerSec.toStringAsFixed(4)} rad/s',
),
],
),
Row(
spacing: 8,
children: [
PrimaryButton(
size: ButtonSize.small,
leading: !_isCalibrated ? SmallProgressIndicator() : null,
onPressed: !_isCalibrated
? null
: () {
// Reset calibration
_isCalibrated = false;
_hasAccelData = false;
_estimator.reset();
_lastGyroUpdate = null;
_lastRoundedAngle = null;
_lastSteeringButton = null;
actionStreamInternal.add(
AlertNotification(LogLevel.LOGLEVEL_INFO, 'Calibrating the sensors now.'),
);
setState(() {});
},
child: Text(_isCalibrated ? 'Calibrate' : 'Calibrating...'),
),
Builder(
builder: (context) {
return PrimaryButton(
size: ButtonSize.small,
trailing: Container(
padding: EdgeInsets.symmetric(horizontal: 4),
decoration: BoxDecoration(
color: Theme.of(context).colorScheme.destructive,
borderRadius: BorderRadius.circular(4),
),
child: Text('${core.settings.getPhoneSteeringThreshold().toInt()}°'),
),
onPressed: () {
final values = [for (var i = 3; i <= 12; i += 1) i];
showDropdown(
context: context,
builder: (b) => DropdownMenu(
children: values
.map(
(v) => MenuButton(
child: Text('$v°'),
onPressed: (c) {
core.settings.setPhoneSteeringThreshold(v);
setState(() {});
},
),
)
.toList(),
),
);
},
child: Text('Trigger Threshold:'),
);
},
),
],
),
if (!_isCalibrated)
Text(
'Calibrating the sensors now. Attach your phone/tablet on your handlebar and keep it still for a second.',
).xSmall,
],
),
);
}
}
class GyroscopeSteeringButtons {
static final ControllerButton leftSteer = ControllerButton(
'gyroLeftSteer',
action: InGameAction.steerLeft,
);
static final ControllerButton rightSteer = ControllerButton(
'gyroRightSteer',
action: InGameAction.steerRight,
);
static List<ControllerButton> get values => [
leftSteer,
rightSteer,
];
}

View File

@@ -0,0 +1,200 @@
import 'dart:math';
/// Pure-Dart steering estimator for phone-on-handlebar steering.
///
/// Design goals:
/// - Avoid long-term drift on platforms like iOS by continuously estimating
/// and subtracting gyro bias.
/// - Keep it testable (no Flutter/sensors dependencies).
///
/// NOTE: This is not a full AHRS. It uses bias-corrected integration and
/// a "stillness" detector to learn gyro bias and optionally auto-recenter.
class SteeringEstimator {
SteeringEstimator({
this.biasLearningRate = 0.02,
this.gyroStillThresholdRadPerSec = 0.03,
this.accelStillThresholdMS2 = 0.6,
this.minStillTimeForBiasSec = 0.35,
this.biasLearningDeadbandDeg = 3.0,
this.minStillTimeForRecenterSec = double.infinity,
this.recenterHalfLifeSec = 0.7,
this.recenterDeadbandDeg = 2.0,
this.maxAngleAbsDeg = 60,
this.lowPassAlpha = 0.9,
// Responsiveness / smoothing tuning.
// When steering changes quickly we reduce smoothing, but keep more
// smoothing when stable to avoid jitter.
this.lowPassAlphaStable = 0.9,
this.lowPassAlphaMoving = 0.55,
this.motionAngleRateDegPerSecForMinAlpha = 90.0,
// Cap dt to avoid "freezing" the estimator on occasional long frames.
this.maxDtSec = 0.05,
});
// Tunables
final double biasLearningRate;
final double gyroStillThresholdRadPerSec;
final double accelStillThresholdMS2;
final double minStillTimeForBiasSec;
final double biasLearningDeadbandDeg;
final double minStillTimeForRecenterSec;
final double recenterHalfLifeSec;
final double recenterDeadbandDeg;
final double maxAngleAbsDeg;
/// Backwards-compatible, kept as-is.
///
/// If you set `lowPassAlpha = 0.0`, filtering is disabled.
final double lowPassAlpha;
/// Smoothing used when the angle is stable.
///
/// Default mirrors the original behavior (`0.9`).
final double lowPassAlphaStable;
/// Smoothing used when the angle is changing quickly.
///
/// Lower alpha => faster response.
final double lowPassAlphaMoving;
/// Angle rate (deg/s) at which we reach `lowPassAlphaMoving`.
final double motionAngleRateDegPerSecForMinAlpha;
/// Maximum timestep used for integration/bias learning.
final double maxDtSec;
// State
double _accelX = 0, _accelY = 0, _accelZ = 0;
bool _hasAccel = false;
double _biasZ = 0.0; // rad/s
double _yawDeg = 0.0;
double _filteredYawDeg = 0.0;
double _stillTimeSec = 0.0;
/// Resets the estimator state.
void reset() {
_biasZ = 0.0;
_yawDeg = 0.0;
_filteredYawDeg = 0.0;
_stillTimeSec = 0.0;
_hasAccel = false;
_accelX = _accelY = _accelZ = 0;
}
/// One-time calibration: assume device is held still and centered.
///
/// This resets yaw and also seeds the bias to the current z gyro rate.
void calibrate({double? seedBiasZRadPerSec}) {
_yawDeg = 0.0;
_filteredYawDeg = 0.0;
_stillTimeSec = 0.0;
if (seedBiasZRadPerSec != null) {
_biasZ = seedBiasZRadPerSec;
}
}
void updateAccel({required double x, required double y, required double z}) {
_accelX = x;
_accelY = y;
_accelZ = z;
_hasAccel = true;
}
/// Update with gyro z-rate (rad/s) and dt (seconds).
///
/// Returns the current filtered steering angle in degrees.
double updateGyro({required double wz, required double dt}) {
if (dt <= 0) {
return angleDeg;
}
// If dt spikes (app paused/jank), cap it instead of bailing out.
// This keeps the estimator responsive and avoids "stuck" output.
final usedDt = dt > maxDtSec ? maxDtSec : dt;
final still = _isStill(wz);
if (still) {
_stillTimeSec += usedDt;
// Learn gyro bias only when we're still AND near our calibrated center.
// Otherwise, if the user holds a steady steering angle, wz≈0 and we'd
// incorrectly move bias towards 0 and cause the angle to be wrong when
// they return to center.
final nearCenter = _yawDeg.abs() <= biasLearningDeadbandDeg;
if (nearCenter && _stillTimeSec >= minStillTimeForBiasSec) {
// Exponential moving average towards the observed rate.
_biasZ = (1.0 - biasLearningRate) * _biasZ + biasLearningRate * wz;
}
// IMPORTANT: only auto-recenter when we're already close to center.
// Users may hold a constant steering angle for several seconds.
final canRecenter = _stillTimeSec >= minStillTimeForRecenterSec && _yawDeg.abs() <= recenterDeadbandDeg;
if (canRecenter) {
_applyRecenter(usedDt);
}
} else {
_stillTimeSec = 0.0;
}
final correctedWz = wz - _biasZ;
_yawDeg += correctedWz * usedDt * (180.0 / pi);
// Clamp to avoid runaway if something goes wrong.
_yawDeg = _yawDeg.clamp(-maxAngleAbsDeg, maxAngleAbsDeg).toDouble();
// Low-pass filter for noise smoothing.
//
// Make it adaptive: when the angle is changing fast, we reduce smoothing
// (more responsive). When stable, we keep stronger smoothing.
//
// If user forces lowPassAlpha=0.0 (existing tests do), we keep behavior
// equivalent (no filtering).
if (lowPassAlpha <= 0.0) {
_filteredYawDeg = _yawDeg;
} else {
final stableAlpha = ((lowPassAlphaStable.isFinite ? lowPassAlphaStable : lowPassAlpha)).clamp(0.0, 0.999);
final movingAlpha = lowPassAlphaMoving.clamp(0.0, stableAlpha);
// Use a rate estimate derived from the filtered-vs-raw divergence.
final rateDegPerSec = ((_yawDeg - _filteredYawDeg).abs()) / usedDt;
final t = (rateDegPerSec / motionAngleRateDegPerSecForMinAlpha).clamp(0.0, 1.0);
final alpha = stableAlpha + (movingAlpha - stableAlpha) * t;
_filteredYawDeg = alpha * _filteredYawDeg + (1 - alpha) * _yawDeg;
}
return angleDeg;
}
double get angleDeg => _filteredYawDeg;
double get biasZRadPerSec => _biasZ;
double get stillTimeSec => _stillTimeSec;
bool _isStill(double wz) {
// If we don't have accel yet, be conservative: don't learn bias.
if (!_hasAccel) return false;
final gyroOk = wz.abs() < gyroStillThresholdRadPerSec;
// Check accel magnitude close to gravity (device not being bumped).
final aMag = sqrt(_accelX * _accelX + _accelY * _accelY + _accelZ * _accelZ);
const g = 9.80665;
final accelOk = (aMag - g).abs() < accelStillThresholdMS2;
return gyroOk && accelOk;
}
void _applyRecenter(double dt) {
// Exponential decay towards 0 with given half-life.
// decay = 0.5^(dt/halfLife)
if (recenterHalfLifeSec <= 0) return;
final decay = pow(0.5, dt / recenterHalfLifeSec).toDouble();
_yawDeg *= decay;
}
}

View File

@@ -1,10 +1,10 @@
import 'package:flutter/material.dart';
import 'package:swift_control/bluetooth/devices/base_device.dart';
import 'package:swift_control/main.dart';
import 'package:swift_control/utils/actions/android.dart';
import 'package:bike_control/bluetooth/devices/base_device.dart';
import 'package:bike_control/utils/actions/android.dart';
import 'package:bike_control/utils/core.dart';
class HidDevice extends BaseDevice {
HidDevice(super.name, {super.availableButtons = const []});
HidDevice(super.name) : super(availableButtons: []);
@override
Future<void> connect() {
@@ -21,11 +21,11 @@ class HidDevice extends BaseDevice {
PopupMenuItem(
child: Text('Ignore'),
onTap: () {
connection.disconnect(this, forget: true);
if (actionHandler is AndroidActions) {
(actionHandler as AndroidActions).ignoreHidDevices();
} else if (connection.isMediaKeyDetectionEnabled.value) {
connection.isMediaKeyDetectionEnabled.value = false;
core.connection.disconnect(this, forget: true, persistForget: true);
if (core.actionHandler is AndroidActions) {
(core.actionHandler as AndroidActions).ignoreHidDevices();
} else if (core.mediaKeyHandler.isMediaKeyDetectionEnabled.value) {
core.mediaKeyHandler.isMediaKeyDetectionEnabled.value = false;
}
},
),

View File

@@ -1,153 +0,0 @@
import 'dart:convert';
import 'dart:io';
import 'package:flutter/foundation.dart';
import 'package:swift_control/utils/keymap/buttons.dart';
import 'package:swift_control/utils/requirements/multi.dart';
class WhooshLink {
Socket? _socket;
ServerSocket? _server;
static final List<InGameAction> supportedActions = [
InGameAction.shiftUp,
InGameAction.shiftDown,
InGameAction.cameraAngle,
InGameAction.emote,
InGameAction.uturn,
InGameAction.steerLeft,
InGameAction.steerRight,
];
final ValueNotifier<bool> isStarted = ValueNotifier(false);
final ValueNotifier<bool> isConnected = ValueNotifier(false);
void stopServer() async {
if (isStarted.value) {
await _socket?.close();
await _server?.close();
isConnected.value = false;
isStarted.value = false;
if (kDebugMode) {
print('Server stopped.');
}
}
}
Future<void> startServer({
required void Function(Socket socket) onConnected,
required void Function(Socket socket) onDisconnected,
}) async {
// Create and bind server socket
_server = await ServerSocket.bind(
InternetAddress.anyIPv6,
21587,
shared: true,
v6Only: false,
);
isStarted.value = true;
if (kDebugMode) {
print('Server started on port ${_server!.port}');
}
// Accept connection
_server!.listen(
(Socket socket) {
_socket = socket;
onConnected(socket);
isConnected.value = true;
if (kDebugMode) {
print('Client connected: ${socket.remoteAddress.address}:${socket.remotePort}');
}
// Listen for data from the client
socket.listen(
(List<int> data) {
try {
if (kDebugMode) {
// TODO we could check if virtual shifting is enabled
final message = utf8.decode(data);
print('Received message: $message');
}
} catch (_) {}
},
onDone: () {
print('Client disconnected: $socket');
onDisconnected(socket);
isConnected.value = false;
},
);
},
);
}
String sendAction(InGameAction action, int? value) {
final jsonObject = switch (action) {
InGameAction.shiftUp => {
'MessageType': 'Controls',
'InGameControls': {
'GearShifting': '1',
},
},
InGameAction.shiftDown => {
'MessageType': 'Controls',
'InGameControls': {
'GearShifting': '-1',
},
},
InGameAction.cameraAngle => {
'MessageType': 'Controls',
'InGameControls': {
'CameraAngle': '$value',
},
},
InGameAction.emote => {
'MessageType': 'Controls',
'InGameControls': {
'Emote': '$value',
},
},
InGameAction.uturn => {
'MessageType': 'Controls',
'InGameControls': {
'UTurn': 'true',
},
},
InGameAction.steerLeft => {
'MessageType': 'Controls',
'InGameControls': {
'Steering': '-1',
},
},
InGameAction.steerRight => {
'MessageType': 'Controls',
'InGameControls': {
'Steering': '1',
},
},
InGameAction.increaseResistance => null,
InGameAction.decreaseResistance => null,
InGameAction.navigateLeft => null,
InGameAction.navigateRight => null,
InGameAction.toggleUi => null,
_ => null,
};
if (jsonObject != null) {
final jsonString = jsonEncode(jsonObject);
_socket?.writeln(jsonString);
return 'Sent action to MyWhoosh: $action ${value ?? ''}';
} else {
return 'No action available for button: $action';
}
}
bool isCompatible(Target target) {
return kIsWeb
? false
: switch (target) {
Target.thisDevice => Platform.isAndroid || Platform.isWindows,
_ => true,
};
}
}

View File

@@ -1,90 +0,0 @@
import 'package:flutter/material.dart';
import 'package:swift_control/bluetooth/devices/base_device.dart';
import 'package:swift_control/main.dart';
import 'package:swift_control/utils/actions/remote.dart';
import 'package:swift_control/widgets/small_progress_indicator.dart';
import 'package:url_launcher/url_launcher_string.dart';
class LinkDevice extends BaseDevice {
String identifier;
LinkDevice(this.identifier) : super('MyWhoosh Direct Connect', availableButtons: []);
@override
Future<void> connect() async {
isConnected = true;
}
@override
Future<void> disconnect() async {
super.disconnect();
whooshLink.stopServer();
isConnected = false;
}
@override
Widget showInformation(BuildContext context) {
return ValueListenableBuilder(
valueListenable: whooshLink.isConnected,
builder: (context, isConnected, _) {
return StatefulBuilder(
builder: (context, setState) {
final myWhooshExplanation = actionHandler is RemoteActions
? 'MyWhoosh Direct Connect allows you to do some additional features such as Emotes and turn directions.'
: 'MyWhoosh Direct Connect is optional, but allows you to do some additional features such as Emotes and turn directions.';
return Row(
children: [
Expanded(
child: SwitchListTile.adaptive(
contentPadding: EdgeInsets.zero,
value: settings.getMyWhooshLinkEnabled(),
onChanged: (value) {
settings.setMyWhooshLinkEnabled(value);
if (!value) {
disconnect();
connection.disconnect(this, forget: true);
} else if (value) {
connection.startMyWhooshServer();
}
setState(() {});
},
title: Text('Enable MyWhoosh Direct Connect'),
subtitle: Row(
spacing: 12,
children: [
if (!settings.getMyWhooshLinkEnabled())
Expanded(
child: Text(
myWhooshExplanation,
style: TextStyle(fontSize: 12),
),
)
else ...[
Expanded(
child: Text(
isConnected ? "Connected" : "Connecting to MyWhoosh...\n$myWhooshExplanation",
style: TextStyle(fontSize: 12),
),
),
if (!isConnected) SmallProgressIndicator(),
],
],
),
),
),
IconButton(
onPressed: () {
launchUrlString('https://www.youtube.com/watch?v=p8sgQhuufeI');
},
icon: Icon(Icons.help_outline),
),
],
);
},
);
},
);
}
}

View File

@@ -0,0 +1,178 @@
import 'dart:convert';
import 'dart:io';
import 'package:bike_control/bluetooth/devices/trainer_connection.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zp.pb.dart';
import 'package:bike_control/bluetooth/messages/notification.dart';
import 'package:bike_control/gen/l10n.dart';
import 'package:bike_control/utils/actions/base_actions.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:bike_control/utils/keymap/keymap.dart';
import 'package:bike_control/utils/requirements/multi.dart';
import 'package:flutter/foundation.dart';
class WhooshLink extends TrainerConnection {
Socket? _socket;
ServerSocket? _server;
static const String connectionTitle = 'MyWhoosh Link';
WhooshLink()
: super(
title: connectionTitle,
supportedActions: [
InGameAction.shiftUp,
InGameAction.shiftDown,
InGameAction.cameraAngle,
InGameAction.emote,
InGameAction.uturn,
InGameAction.steerLeft,
InGameAction.steerRight,
],
);
void stopServer() async {
await _socket?.close();
await _server?.close();
isConnected.value = false;
isStarted.value = false;
if (kDebugMode) {
print('Server stopped.');
}
}
Future<void> startServer() async {
isStarted.value = true;
try {
// Create and bind server socket
_server = await ServerSocket.bind(
InternetAddress.anyIPv6,
21587,
shared: true,
v6Only: false,
);
} catch (e) {
if (kDebugMode) {
print('Failed to start server: $e');
}
isConnected.value = false;
isStarted.value = false;
rethrow;
}
if (kDebugMode) {
print('Server started on port ${_server!.port}');
}
// Accept connection
_server!.listen(
(Socket socket) {
_socket = socket;
core.connection.signalNotification(
AlertNotification(LogLevel.LOGLEVEL_INFO, AppLocalizations.current.myWhooshLinkConnected),
);
isConnected.value = true;
if (kDebugMode) {
print('Client connected: ${socket.remoteAddress.address}:${socket.remotePort}');
}
// Listen for data from the client
socket.listen(
(List<int> data) {
try {
if (kDebugMode) {
// TODO we could check if virtual shifting is enabled
final message = utf8.decode(data);
print('Received message: $message');
}
} catch (_) {}
},
onDone: () {
print('Client disconnected: $socket');
isConnected.value = false;
core.connection.signalNotification(
AlertNotification(LogLevel.LOGLEVEL_WARNING, 'MyWhoosh Link disconnected'),
);
},
);
},
);
}
@override
Future<ActionResult> sendAction(KeyPair keyPair, {required bool isKeyDown, required bool isKeyUp}) async {
final jsonObject = switch (keyPair.inGameAction) {
InGameAction.shiftUp => {
'MessageType': 'Controls',
'InGameControls': {
'GearShifting': '1',
},
},
InGameAction.shiftDown => {
'MessageType': 'Controls',
'InGameControls': {
'GearShifting': '-1',
},
},
InGameAction.cameraAngle => {
'MessageType': 'Controls',
'InGameControls': {
'CameraAngle': '${keyPair.inGameActionValue}',
},
},
InGameAction.emote => {
'MessageType': 'Controls',
'InGameControls': {
'Emote': '${keyPair.inGameActionValue}',
},
},
InGameAction.uturn => {
'MessageType': 'Controls',
'InGameControls': {
'UTurn': 'true',
},
},
InGameAction.steerLeft => {
'MessageType': 'Controls',
'InGameControls': {
'Steering': isKeyDown ? '-1' : '0',
},
},
InGameAction.steerRight => {
'MessageType': 'Controls',
'InGameControls': {
'Steering': isKeyDown ? '1' : '0',
},
},
InGameAction.increaseResistance => null,
InGameAction.decreaseResistance => null,
InGameAction.navigateLeft => null,
InGameAction.navigateRight => null,
InGameAction.toggleUi => null,
_ => null,
};
final supportsIsKeyUpActions = [
InGameAction.steerLeft,
InGameAction.steerRight,
];
if (jsonObject != null && !isKeyDown && !supportsIsKeyUpActions.contains(keyPair.inGameAction)) {
return Ignored('No Action sent on key down for action: ${keyPair.inGameAction}');
} else if (jsonObject != null) {
final jsonString = jsonEncode(jsonObject);
_socket?.writeln(jsonString);
return Success('Sent action to MyWhoosh: ${keyPair.inGameAction} ${keyPair.inGameActionValue ?? ''}');
} else {
return NotHandled('No action available for button: ${keyPair.inGameAction}');
}
}
bool isCompatible(Target target) {
return kIsWeb
? false
: switch (target) {
Target.thisDevice => !Platform.isWindows,
_ => true,
};
}
}

View File

@@ -0,0 +1,267 @@
import 'dart:io';
import 'package:bike_control/bluetooth/devices/openbikecontrol/openbikecontrol_device.dart';
import 'package:bike_control/bluetooth/devices/openbikecontrol/protocol_parser.dart';
import 'package:bike_control/bluetooth/devices/trainer_connection.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zp.pbenum.dart';
import 'package:bike_control/utils/actions/base_actions.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:bike_control/utils/keymap/keymap.dart';
import 'package:bike_control/widgets/title.dart';
import 'package:bluetooth_low_energy/bluetooth_low_energy.dart';
import 'package:dartx/dartx.dart';
import 'package:flutter/foundation.dart';
import '../../messages/notification.dart' show AlertNotification;
class OpenBikeControlBluetoothEmulator extends TrainerConnection {
late final _peripheralManager = PeripheralManager();
final ValueNotifier<AppInfo?> connectedApp = ValueNotifier<AppInfo?>(null);
bool _isServiceAdded = false;
bool _isSubscribedToEvents = false;
Central? _central;
late GATTCharacteristic _buttonCharacteristic;
static const String connectionTitle = 'OpenBikeControl BLE Emulator';
OpenBikeControlBluetoothEmulator()
: super(
title: connectionTitle,
supportedActions: InGameAction.values,
);
Future<void> startServer() async {
isStarted.value = true;
_peripheralManager.stateChanged.forEach((state) {
print('Peripheral manager state: ${state.state}');
});
if (!kIsWeb && Platform.isAndroid) {
_peripheralManager.connectionStateChanged.forEach((state) {
print('Peripheral connection state: ${state.state} of ${state.central.uuid}');
if (state.state == ConnectionState.connected) {
} else if (state.state == ConnectionState.disconnected) {
if (connectedApp.value != null) {
core.connection.signalNotification(
AlertNotification(LogLevel.LOGLEVEL_INFO, 'Disconnected from app: ${connectedApp.value?.appId}'),
);
}
isConnected.value = false;
connectedApp.value = null;
_central = null;
}
});
}
while (_peripheralManager.state != BluetoothLowEnergyState.poweredOn && core.settings.getObpBleEnabled()) {
print('Waiting for peripheral manager to be powered on...');
await Future.delayed(Duration(seconds: 1));
}
_buttonCharacteristic = GATTCharacteristic.mutable(
uuid: UUID.fromString(OpenBikeControlConstants.BUTTON_STATE_CHARACTERISTIC_UUID),
descriptors: [],
properties: [
GATTCharacteristicProperty.notify,
],
permissions: [],
);
if (!_isServiceAdded) {
await Future.delayed(Duration(seconds: 1));
if (!_isSubscribedToEvents) {
_isSubscribedToEvents = true;
_peripheralManager.characteristicReadRequested.forEach((eventArgs) async {
print('Read request for characteristic: ${eventArgs.characteristic.uuid}');
switch (eventArgs.characteristic.uuid.toString().toUpperCase()) {
default:
print('Unhandled read request for characteristic: ${eventArgs.characteristic.uuid}');
}
final request = eventArgs.request;
final trimmedValue = Uint8List.fromList([]);
await _peripheralManager.respondReadRequestWithValue(
request,
value: trimmedValue,
);
// You can respond to read requests here if needed
});
_peripheralManager.characteristicNotifyStateChanged.forEach((char) {
_central = char.central;
print(
'Notify state changed for characteristic: ${char.characteristic.uuid}: ${char.state}',
);
});
_peripheralManager.characteristicWriteRequested.forEach((eventArgs) async {
final characteristic = eventArgs.characteristic;
final request = eventArgs.request;
final value = request.value;
print(
'Write request for characteristic: ${characteristic.uuid}',
);
switch (eventArgs.characteristic.uuid.toString().toLowerCase()) {
case OpenBikeControlConstants.APPINFO_CHARACTERISTIC_UUID:
try {
final appInfo = OpenBikeProtocolParser.parseAppInfo(value);
isConnected.value = true;
connectedApp.value = appInfo;
supportedActions = appInfo.supportedButtons.mapNotNull((b) => b.action).toList();
core.connection.signalNotification(
AlertNotification(LogLevel.LOGLEVEL_INFO, 'Connected to app: ${appInfo.appId}'),
);
print('Parsed App Info: $appInfo');
} catch (e) {
print('Error parsing App Info: $e');
}
break;
default:
print('Unhandled write request for characteristic: ${eventArgs.characteristic.uuid}');
}
await _peripheralManager.respondWriteRequest(request);
});
}
if (!Platform.isWindows) {
// Device Information
await _peripheralManager.addService(
GATTService(
uuid: UUID.fromString('180A'),
isPrimary: true,
characteristics: [
GATTCharacteristic.immutable(
uuid: UUID.fromString('2A29'),
value: Uint8List.fromList('BikeControl'.codeUnits),
descriptors: [],
),
GATTCharacteristic.immutable(
uuid: UUID.fromString('2A25'),
value: Uint8List.fromList('1337'.codeUnits),
descriptors: [],
),
GATTCharacteristic.immutable(
uuid: UUID.fromString('2A27'),
value: Uint8List.fromList('1.0'.codeUnits),
descriptors: [],
),
GATTCharacteristic.immutable(
uuid: UUID.fromString('2A26'),
value: Uint8List.fromList((packageInfoValue?.version ?? '1.0.0').codeUnits),
descriptors: [],
),
],
includedServices: [],
),
);
}
// Battery Service
await _peripheralManager.addService(
GATTService(
uuid: UUID.fromString('180F'),
isPrimary: true,
characteristics: [
GATTCharacteristic.mutable(
uuid: UUID.fromString('2A19'),
descriptors: [],
properties: [
GATTCharacteristicProperty.read,
GATTCharacteristicProperty.notify,
],
permissions: [
GATTCharacteristicPermission.read,
],
),
],
includedServices: [],
),
);
// Unknown Service
await _peripheralManager.addService(
GATTService(
uuid: UUID.fromString(OpenBikeControlConstants.SERVICE_UUID),
isPrimary: true,
characteristics: [
_buttonCharacteristic,
GATTCharacteristic.mutable(
uuid: UUID.fromString(OpenBikeControlConstants.APPINFO_CHARACTERISTIC_UUID),
descriptors: [],
properties: [
GATTCharacteristicProperty.writeWithoutResponse,
GATTCharacteristicProperty.write,
],
permissions: [
GATTCharacteristicPermission.read,
GATTCharacteristicPermission.write,
],
),
],
includedServices: [],
),
);
_isServiceAdded = true;
}
final advertisement = Advertisement(
name: 'BikeControl',
serviceUUIDs: [UUID.fromString(OpenBikeControlConstants.SERVICE_UUID)],
);
print('Starting advertising with OpenBikeControl service...');
await _peripheralManager.startAdvertising(advertisement);
}
Future<void> stopServer() async {
if (kDebugMode) {
print('Stopping OpenBikeControl BLE server...');
}
await _peripheralManager.stopAdvertising();
isStarted.value = false;
isConnected.value = false;
connectedApp.value = null;
}
@override
Future<ActionResult> sendAction(KeyPair keyPair, {required bool isKeyDown, required bool isKeyUp}) async {
final inGameAction = keyPair.inGameAction;
final mappedButtons = connectedApp.value!.supportedButtons.filter(
(supportedButton) => supportedButton.action == inGameAction,
);
if (inGameAction == null) {
return Error('Invalid in-game action for key pair: $keyPair');
} else if (_central == null) {
return Error('No central connected');
} else if (connectedApp.value == null) {
return Error('No app info received from central');
} else if (mappedButtons.isEmpty) {
return NotHandled('App does not support all buttons for action: ${inGameAction.title}');
}
if (isKeyDown && isKeyUp) {
final responseDataDown = OpenBikeProtocolParser.encodeButtonState(
mappedButtons.map((b) => ButtonState(b, 1)).toList(),
);
await _peripheralManager.notifyCharacteristic(_central!, _buttonCharacteristic, value: responseDataDown);
final responseDataUp = OpenBikeProtocolParser.encodeButtonState(
mappedButtons.map((b) => ButtonState(b, 0)).toList(),
);
await _peripheralManager.notifyCharacteristic(_central!, _buttonCharacteristic, value: responseDataUp);
} else {
final responseData = OpenBikeProtocolParser.encodeButtonState(
mappedButtons.map((b) => ButtonState(b, isKeyDown ? 1 : 0)).toList(),
);
await _peripheralManager.notifyCharacteristic(_central!, _buttonCharacteristic, value: responseData);
}
return Success('Buttons ${inGameAction.title} sent');
}
}

View File

@@ -0,0 +1,207 @@
import 'dart:io';
import 'package:bike_control/bluetooth/devices/openbikecontrol/openbikecontrol_device.dart';
import 'package:bike_control/bluetooth/devices/openbikecontrol/protocol_parser.dart';
import 'package:bike_control/bluetooth/devices/trainer_connection.dart';
import 'package:bike_control/bluetooth/devices/zwift/ftms_mdns_emulator.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zp.pb.dart';
import 'package:bike_control/bluetooth/messages/notification.dart';
import 'package:bike_control/utils/actions/base_actions.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:bike_control/utils/keymap/keymap.dart';
import 'package:dartx/dartx.dart';
import 'package:flutter/foundation.dart';
import 'package:nsd/nsd.dart';
class OpenBikeControlMdnsEmulator extends TrainerConnection {
ServerSocket? _server;
Registration? _mdnsRegistration;
static const String connectionTitle = 'OpenBikeControl mDNS Emulator';
final ValueNotifier<AppInfo?> connectedApp = ValueNotifier(null);
Socket? _socket;
OpenBikeControlMdnsEmulator()
: super(
title: connectionTitle,
supportedActions: InGameAction.values,
);
Future<void> startServer() async {
print('Starting mDNS server...');
isStarted.value = true;
// Get local IP
final interfaces = await NetworkInterface.list();
InternetAddress? localIP;
for (final interface in interfaces) {
for (final addr in interface.addresses) {
if (addr.type == InternetAddressType.IPv4 && !addr.isLoopback) {
localIP = addr;
break;
}
}
if (localIP != null) break;
}
if (localIP == null) {
throw 'Could not find network interface';
}
_createTcpServer();
if (kDebugMode) {
enableLogging(LogTopic.calls);
enableLogging(LogTopic.errors);
}
disableServiceTypeValidation(true);
try {
// Create service
_mdnsRegistration = await register(
Service(
name: 'BikeControl',
type: '_openbikecontrol._tcp',
port: 36867,
//hostName: 'KICKR BIKE SHIFT B84D.local',
addresses: [localIP],
txt: {
'version': Uint8List.fromList([0x01]),
'id': Uint8List.fromList('1337'.codeUnits),
'name': Uint8List.fromList('BikeControl'.codeUnits),
'service-uuids': Uint8List.fromList(OpenBikeControlConstants.SERVICE_UUID.codeUnits),
'manufacturer': Uint8List.fromList('OpenBikeControl'.codeUnits),
'model': Uint8List.fromList('BikeControl app'.codeUnits),
},
),
);
print('Service: ${_mdnsRegistration!.id} at ${localIP.address}:$_mdnsRegistration');
print('Server started - advertising service!');
} catch (e, s) {
core.connection.signalNotification(AlertNotification(LogLevel.LOGLEVEL_ERROR, 'Failed to start mDNS server: $e'));
rethrow;
}
}
Future<void> stopServer() async {
if (kDebugMode) {
print('Stopping OpenBikeControl mDNS server...');
}
if (_mdnsRegistration != null) {
unregister(_mdnsRegistration!);
_mdnsRegistration = null;
}
isStarted.value = false;
isConnected.value = false;
connectedApp.value = null;
_socket?.destroy();
_socket = null;
}
Future<void> _createTcpServer() async {
try {
_server = await ServerSocket.bind(
InternetAddress.anyIPv6,
36867,
shared: true,
v6Only: false,
);
} catch (e) {
core.connection.signalNotification(AlertNotification(LogLevel.LOGLEVEL_ERROR, 'Failed to start server: $e'));
rethrow;
}
if (kDebugMode) {
print('Server started on port ${_server!.port}');
}
// Accept connection
_server!.listen(
(Socket socket) {
_socket = socket;
if (kDebugMode) {
print('Client connected: ${socket.remoteAddress.address}:${socket.remotePort}');
}
// Listen for data from the client
socket.listen(
(List<int> data) {
if (kDebugMode) {
print('Received message: ${bytesToHex(data)}');
}
final messageType = data[0];
switch (messageType) {
case OpenBikeProtocolParser.MSG_TYPE_APP_INFO:
final appInfo = OpenBikeProtocolParser.parseAppInfo(Uint8List.fromList(data));
isConnected.value = true;
connectedApp.value = appInfo;
supportedActions = appInfo.supportedButtons.mapNotNull((b) => b.action).toList();
core.connection.signalNotification(
AlertNotification(LogLevel.LOGLEVEL_INFO, 'Connected to app: ${appInfo.appId}'),
);
break;
default:
print('Unknown message type: $messageType');
}
},
onDone: () {
core.connection.signalNotification(
AlertNotification(LogLevel.LOGLEVEL_INFO, 'Disconnected from app: ${connectedApp.value?.appId}'),
);
isConnected.value = false;
connectedApp.value = null;
_socket = null;
},
);
},
);
}
@override
Future<ActionResult> sendAction(KeyPair keyPair, {required bool isKeyDown, required bool isKeyUp}) async {
final inGameAction = keyPair.inGameAction;
final mappedButtons = connectedApp.value!.supportedButtons.filter(
(supportedButton) => supportedButton.action == inGameAction,
);
if (inGameAction == null) {
return Error('Invalid in-game action for key pair: $keyPair');
} else if (_socket == null) {
print('No client connected, cannot send button press');
return Error('No client connected');
} else if (connectedApp.value == null) {
return Error('No app info received from central');
} else if (mappedButtons.isEmpty) {
return NotHandled('App does not support: ${inGameAction.title}');
}
if (isKeyDown && isKeyUp) {
final responseDataDown = OpenBikeProtocolParser.encodeButtonState(
mappedButtons.map((b) => ButtonState(b, 1)).toList(),
);
_write(_socket!, responseDataDown);
final responseDataUp = OpenBikeProtocolParser.encodeButtonState(
mappedButtons.map((b) => ButtonState(b, 0)).toList(),
);
_write(_socket!, responseDataUp);
} else {
final responseData = OpenBikeProtocolParser.encodeButtonState(
mappedButtons.map((b) => ButtonState(b, isKeyDown ? 1 : 0)).toList(),
);
_write(_socket!, responseData);
}
return Success('Sent ${inGameAction.title} button press');
}
void _write(Socket socket, List<int> responseData) {
debugPrint('Sending response: ${bytesToHex(responseData)}');
socket.add(responseData);
}
}

View File

@@ -0,0 +1,80 @@
import 'dart:typed_data';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zp.pb.dart';
import 'package:bike_control/bluetooth/messages/notification.dart';
import 'package:bike_control/widgets/title.dart';
import 'package:universal_ble/universal_ble.dart';
import '../bluetooth_device.dart';
import 'protocol_parser.dart';
class OpenBikeControlDevice extends BluetoothDevice {
OpenBikeControlDevice(super.scanResult)
: super(
availableButtons: OpenBikeProtocolParser.BUTTON_NAMES.values.toList(),
);
@override
Future<void> handleServices(List<BleService> services) async {
final service = services.firstWhere(
(e) => e.uuid.toLowerCase() == OpenBikeControlConstants.SERVICE_UUID.toLowerCase(),
orElse: () => throw Exception('Service not found: ${OpenBikeControlConstants.SERVICE_UUID}'),
);
final characteristic = service.characteristics.firstWhere(
(e) => e.uuid.toLowerCase() == OpenBikeControlConstants.BUTTON_STATE_CHARACTERISTIC_UUID.toLowerCase(),
orElse: () =>
throw Exception('Characteristic not found: ${OpenBikeControlConstants.BUTTON_STATE_CHARACTERISTIC_UUID}'),
);
await UniversalBle.subscribeNotifications(device.deviceId, service.uuid, characteristic.uuid);
final appInfoCharacteristic = service.characteristics.firstWhere(
(e) => e.uuid.toLowerCase() == OpenBikeControlConstants.APPINFO_CHARACTERISTIC_UUID.toLowerCase(),
orElse: () =>
throw Exception('Characteristic not found: ${OpenBikeControlConstants.APPINFO_CHARACTERISTIC_UUID}'),
);
await UniversalBle.write(
device.deviceId,
service.uuid,
appInfoCharacteristic.uuid,
OpenBikeProtocolParser.encodeAppInfo(
appId: 'BikeControl',
appVersion: packageInfoValue!.version,
supportedButtons: OpenBikeProtocolParser.BUTTON_NAMES.values.toList(),
),
);
}
@override
Future<void> processCharacteristic(String characteristic, Uint8List bytes) async {
final charLower = characteristic.toLowerCase();
if (charLower == OpenBikeControlConstants.BUTTON_STATE_CHARACTERISTIC_UUID.toLowerCase()) {
try {
final parsed = OpenBikeProtocolParser.parseButtonState(bytes);
final buttonsToPress = parsed.where((e) => e.state == 1).map((e) => e.button).toList();
await handleButtonsClicked(buttonsToPress);
} catch (e) {
actionStreamInternal.add(AlertNotification(LogLevel.LOGLEVEL_WARNING, 'Error parsing OpenBike message: $e'));
}
}
}
}
class OpenBikeControlConstants {
// OpenBikeControl BLE service and characteristic UUIDs (see BLE.md)
static const String SERVICE_UUID = 'd273f680-d548-419d-b9d1-fa0472345229';
// Button State Characteristic (Notify)
static const String BUTTON_STATE_CHARACTERISTIC_UUID = 'd273f681-d548-419d-b9d1-fa0472345229';
// Haptic Feedback Characteristic (Write)
static const String HAPTIC_CHARACTERISTIC_UUID = 'd273f682-d548-419d-b9d1-fa0472345229';
// App Info Characteristic (Write)
static const String APPINFO_CHARACTERISTIC_UUID = 'd273f683-d548-419d-b9d1-fa0472345229';
}

View File

@@ -0,0 +1,308 @@
// OpenBikeControl Protocol Parser (Dart)
// This file is a translation of the Python `protocol_parser.py` example into Dart.
// It provides simple encoding/decoding utilities for the OpenBikeControl message
// types used in the Python example. This is intentionally a small, focused
// module that mirrors the original Python API.
import 'dart:convert';
import 'dart:typed_data';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:dartx/dartx.dart';
class ProtocolParseException implements Exception {
final String message;
final Uint8List? raw;
ProtocolParseException(this.message, [this.raw]);
@override
String toString() => 'ProtocolParseException: $message${raw != null ? ' raw=${raw!.length}' : ''}';
}
class OpenBikeProtocolParser {
// Button ID to name mapping (based on PROTOCOL.md in Python example)
static const Map<int, ControllerButton> BUTTON_NAMES = {
// Gear Shifting (0x01-0x0F)
0x01: ControllerButton('Shift Up', identifier: 0x01, action: InGameAction.shiftUp),
0x02: ControllerButton('Shift Down', identifier: 0x02, action: InGameAction.shiftDown),
0x03: ControllerButton('Gear Set', identifier: 0x03),
// Navigation (0x10-0x1F)
0x10: ControllerButton('Up/Steer Left', identifier: 0x10, action: InGameAction.steerLeft),
0x11: ControllerButton('Down/Steer Right', identifier: 0x11, action: InGameAction.steerRight),
0x12: ControllerButton('Left/Look Left', identifier: 0x12, action: InGameAction.navigateLeft),
0x13: ControllerButton('Right/Look Right', identifier: 0x13, action: InGameAction.navigateRight),
0x14: ControllerButton('Select/Confirm', identifier: 0x14, action: InGameAction.select),
0x15: ControllerButton('Back/Cancel', identifier: 0x15, action: InGameAction.back),
0x16: ControllerButton('Menu', identifier: 0x16),
0x17: ControllerButton('Home', identifier: 0x17),
// Social/Emotes (0x20-0x2F)
0x20: ControllerButton('Wave', identifier: 0x20, action: InGameAction.emote),
0x21: ControllerButton('Thumbs Up', identifier: 0x21, action: InGameAction.emote),
0x22: ControllerButton('Hammer Time', identifier: 0x22, action: InGameAction.emote),
0x23: ControllerButton('Bell', identifier: 0x23),
0x24: ControllerButton('Screenshot', identifier: 0x24),
// Training Controls (0x30-0x3F)
0x30: ControllerButton('ERG Up', identifier: 0x30, action: InGameAction.increaseResistance),
0x31: ControllerButton('ERG Down', identifier: 0x31, action: InGameAction.decreaseResistance),
0x32: ControllerButton('Skip Interval', identifier: 0x32),
0x33: ControllerButton('Pause', identifier: 0x33),
0x34: ControllerButton('Resume', identifier: 0x34),
0x35: ControllerButton('Lap', identifier: 0x35),
// View Controls (0x40-0x4F)
0x40: ControllerButton('Camera Angle', identifier: 0x40, action: InGameAction.cameraAngle),
0x41: ControllerButton('Camera 1', identifier: 0x41, action: InGameAction.cameraAngle),
0x42: ControllerButton('Camera 2', identifier: 0x42, action: InGameAction.cameraAngle),
0x43: ControllerButton('Camera 3', identifier: 0x43, action: InGameAction.cameraAngle),
0x44: ControllerButton('HUD Toggle', identifier: 0x44, action: InGameAction.toggleUi),
0x45: ControllerButton('Map Toggle', identifier: 0x45),
// Power-ups (0x50-0x5F)
0x50: ControllerButton('Power-up 1', identifier: 0x50, action: InGameAction.usePowerUp),
0x51: ControllerButton('Power-up 2', identifier: 0x51, action: InGameAction.usePowerUp),
0x52: ControllerButton('Power-up 3', identifier: 0x52, action: InGameAction.usePowerUp),
};
// Haptic feedback patterns
static const Map<String, int> HAPTIC_PATTERNS = {
'none': 0x00,
'short': 0x01,
'double': 0x02,
'triple': 0x03,
'long': 0x04,
'success': 0x05,
'warning': 0x06,
'error': 0x07,
};
// Message types (for TCP/mDNS protocol)
static const int MSG_TYPE_BUTTON_STATE = 0x01;
static const int MSG_TYPE_DEVICE_STATUS = 0x02;
static const int MSG_TYPE_HAPTIC_FEEDBACK = 0x03;
static const int MSG_TYPE_APP_INFO = 0x04;
/// Parse button state data from binary format.
/// Data format: [Message_Type, Button_ID_1, State_1, Button_ID_2, State_2, ...]
static List<ButtonState> parseButtonState(Uint8List data) {
final buttons = <ButtonState>[];
if (data.isEmpty) return buttons;
if (data[0] != MSG_TYPE_BUTTON_STATE) return buttons;
for (var i = 1; i < data.length; i += 2) {
if (i + 1 < data.length) {
final buttonId = data[i];
final state = data[i + 1];
if (BUTTON_NAMES[buttonId] != null) {
buttons.add(ButtonState(BUTTON_NAMES[buttonId]!, state));
} else {
throw ProtocolParseException('Unknown button ID: 0x${buttonId.toRadixString(16).padLeft(2, '0')}', data);
}
}
}
return buttons;
}
static Uint8List encodeButtonState(List<ButtonState> buttons) {
final bytes = BytesBuilder();
bytes.addByte(MSG_TYPE_BUTTON_STATE);
for (final b in buttons) {
bytes.addByte(b.button.identifier!);
bytes.addByte(b.state);
}
return bytes.toBytes();
}
static DeviceStatus parseDeviceStatus(Uint8List data) {
if (data.length < 3) {
throw ProtocolParseException('Device status message too short', data);
}
if (data[0] != MSG_TYPE_DEVICE_STATUS) {
throw ProtocolParseException('Invalid message type: ${data[0]}, expected $MSG_TYPE_DEVICE_STATUS', data);
}
final battery = data[1] == 0xFF ? null : data[1];
final connected = data[2] == 0x01;
return DeviceStatus(battery: battery, connected: connected);
}
static Uint8List encodeDeviceStatus({int? battery, bool connected = true}) {
final batteryByte = battery == null ? 0xFF : (battery & 0xFF);
final connectedByte = connected ? 0x01 : 0x00;
return Uint8List.fromList([MSG_TYPE_DEVICE_STATUS, batteryByte, connectedByte]);
}
static Uint8List encodeHapticFeedback({String pattern = 'short', int duration = 0, int intensity = 0}) {
final patternByte = HAPTIC_PATTERNS[pattern] ?? HAPTIC_PATTERNS['short']!;
final bytes = Uint8List(4);
bytes[0] = MSG_TYPE_HAPTIC_FEEDBACK;
bytes[1] = patternByte;
bytes[2] = duration & 0xFF;
bytes[3] = intensity & 0xFF;
return bytes;
}
static HapticFeedbackMessage parseHapticFeedback(Uint8List data) {
if (data.length < 4) {
throw ProtocolParseException('Haptic feedback message too short', data);
}
if (data[0] != MSG_TYPE_HAPTIC_FEEDBACK) {
throw ProtocolParseException('Invalid message type: ${data[0]}', data);
}
final patternByte = data[1];
final duration = data[2];
final intensity = data[3];
String patternName = 'unknown';
HAPTIC_PATTERNS.forEach((name, value) {
if (value == patternByte) patternName = name;
});
return HapticFeedbackMessage(
pattern: patternName,
patternByte: patternByte,
duration: duration,
intensity: intensity,
);
}
static Uint8List encodeAppInfo({
required String appId,
required String appVersion,
required List<ControllerButton> supportedButtons,
}) {
final appIdBytes = utf8.encode(appId).take(32).toList();
final appVersionBytes = utf8.encode(appVersion).take(32).toList();
final builder = BytesBuilder();
builder.addByte(MSG_TYPE_APP_INFO);
builder.addByte(0x01); // Version
builder.addByte(appIdBytes.length);
builder.add(appIdBytes);
builder.addByte(appVersionBytes.length);
builder.add(appVersionBytes);
builder.addByte(supportedButtons.length);
builder.add(supportedButtons.map((e) => e.identifier!).toList());
return builder.toBytes();
}
static AppInfo parseAppInfo(Uint8List data) {
if (data.isEmpty || data[0] != MSG_TYPE_APP_INFO) {
throw ProtocolParseException('Invalid message type', data);
}
var idx = 1;
if (data.length < idx + 3) {
throw ProtocolParseException('App info message too short', data);
}
final version = data[idx];
idx += 1;
if (version != 0x01) {
throw ProtocolParseException('Unsupported app info version: $version', data);
}
if (idx >= data.length) throw ProtocolParseException('Missing app ID length', data);
final appIdLen = data[idx];
idx += 1;
if (idx + appIdLen > data.length) throw ProtocolParseException('App ID length exceeds buffer', data);
final appId = utf8.decode(data.sublist(idx, idx + appIdLen));
idx += appIdLen;
if (idx >= data.length) throw ProtocolParseException('Missing app version length', data);
final appVersionLen = data[idx];
idx += 1;
if (idx + appVersionLen > data.length) throw ProtocolParseException('App version length exceeds buffer', data);
final appVersion = utf8.decode(data.sublist(idx, idx + appVersionLen));
idx += appVersionLen;
if (idx >= data.length) throw ProtocolParseException('Missing button count', data);
final buttonCount = data[idx];
idx += 1;
if (idx + buttonCount > data.length) throw ProtocolParseException('Button count exceeds buffer', data);
final buttonIds = data.sublist(idx, idx + buttonCount).toList();
final controllerButtons = buttonIds.mapNotNull((id) => BUTTON_NAMES[id]).toList();
return AppInfo(
appId: appId,
appVersion: appVersion,
supportedButtons: controllerButtons,
supportedActions: controllerButtons.mapNotNull((b) => b.action).toList(),
);
}
}
class AppInfo {
final String appId;
final String appVersion;
final List<ControllerButton> supportedButtons;
final List<InGameAction> supportedActions;
AppInfo({
required this.appId,
required this.appVersion,
required this.supportedButtons,
required this.supportedActions,
});
@override
String toString() =>
'AppInfo(appId: $appId, appVersion: $appVersion, supportedButtons: $supportedButtons, supportedActions: $supportedActions)';
}
/// DeviceStatus message representation
class DeviceStatus {
final int? battery; // 0-100, null if 0xFF
final bool connected;
DeviceStatus({required this.battery, required this.connected});
@override
String toString() => 'DeviceStatus(battery: ${battery ?? 'unknown'}, connected: $connected)';
}
class ButtonState {
/// Represents a single button id/state pair.class ButtonState {
final ControllerButton button;
final int state; // 0=released,1=pressed,2-255=analog
const ButtonState(this.button, this.state);
@override
String toString() => formatButtonState(button, state);
String formatButtonState(ControllerButton button, int state) {
final buttonName = button.name;
String stateStr;
if (state == 0) {
stateStr = 'RELEASED';
} else if (state == 1) {
stateStr = 'PRESSED';
} else {
final percentage = ((state - 2) / (255 - 2) * 100).round();
stateStr = 'ANALOG $percentage%';
}
return '$buttonName: $stateStr';
}
}
/// Haptic feedback representation
class HapticFeedbackMessage {
final String pattern;
final int patternByte;
final int duration; // in 10ms units
final int intensity; // 0-255
HapticFeedbackMessage({
required this.pattern,
required this.patternByte,
required this.duration,
required this.intensity,
});
@override
String toString() => 'HapticFeedback(pattern: $pattern, duration: $duration, intensity: $intensity)';
}

View File

@@ -1,10 +1,10 @@
import 'dart:typed_data';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/keymap/apps/custom_app.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:dartx/dartx.dart';
import 'package:flutter/material.dart';
import 'package:swift_control/main.dart';
import 'package:swift_control/utils/keymap/apps/custom_app.dart';
import 'package:swift_control/utils/keymap/buttons.dart';
import 'package:universal_ble/universal_ble.dart';
import '../bluetooth_device.dart';
@@ -30,7 +30,7 @@ class ShimanoDi2 extends BluetoothDevice {
bool _isInitialized = false;
@override
Future<void> processCharacteristic(String characteristic, Uint8List bytes) {
Future<void> processCharacteristic(String characteristic, Uint8List bytes) async {
if (characteristic.toLowerCase() == ShimanoDi2Constants.D_FLY_CHANNEL_UUID) {
final channels = bytes.sublist(1);
@@ -40,7 +40,7 @@ class ShimanoDi2 extends BluetoothDevice {
final readableIndex = index + 1;
_lastButtons[index] = value;
actionHandler.supportedApp?.keymap.getOrAddButton(
getOrAddButton(
'D-Fly Channel $readableIndex',
() => ControllerButton('D-Fly Channel $readableIndex'),
);
@@ -57,18 +57,17 @@ class ShimanoDi2 extends BluetoothDevice {
final readableIndex = index + 1;
final button = actionHandler.supportedApp?.keymap.getOrAddButton(
final button = getOrAddButton(
'D-Fly Channel $readableIndex',
() => ControllerButton('D-Fly Channel $readableIndex'),
);
if (didChange && button != null) {
if (didChange) {
clickedButtons.add(button);
}
});
if (clickedButtons.isNotEmpty) {
handleButtonsClicked(clickedButtons);
handleButtonsClicked([]);
await handleButtonsClickedWithoutLongPressSupport(clickedButtons);
}
}
return Future.value();
@@ -84,7 +83,7 @@ class ShimanoDi2 extends BluetoothDevice {
'Make sure to set your Di2 buttons to D-Fly channels in the Shimano E-TUBE app.',
style: TextStyle(fontSize: 12, color: Colors.grey),
),
if (actionHandler.supportedApp is! CustomApp)
if (core.actionHandler.supportedApp is! CustomApp)
Text(
'Use a custom keymap to support ${scanResult.name}',
style: TextStyle(fontSize: 12, color: Colors.grey),
@@ -96,6 +95,7 @@ class ShimanoDi2 extends BluetoothDevice {
class ShimanoDi2Constants {
static const String SERVICE_UUID = "000018ef-5348-494d-414e-4f5f424c4500";
static const String SERVICE_UUID_ALTERNATIVE = "000018ff-5348-494d-414e-4f5f424c4500";
static const String D_FLY_CHANNEL_UUID = "00002ac2-5348-494d-414e-4f5f424c4500";
}

View File

@@ -0,0 +1,171 @@
import 'dart:async';
import 'package:bike_control/bluetooth/devices/zwift/ftms_mdns_emulator.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:flutter/foundation.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
import 'package:universal_ble/universal_ble.dart';
import '../bluetooth_device.dart';
class SramAxs extends BluetoothDevice {
SramAxs(super.scanResult) : super(availableButtons: [], isBeta: true);
Timer? _singleClickTimer;
int _tapCount = 0;
@override
Future<void> disconnect() async {
_singleClickTimer?.cancel();
_singleClickTimer = null;
_tapCount = 0;
await super.disconnect();
}
@override
Future<void> handleServices(List<BleService> services) async {
final service = services.firstWhere(
(e) => e.uuid.toLowerCase() == SramAxsConstants.SERVICE_UUID_RELEVANT.toLowerCase(),
orElse: () => throw Exception('Service not found: ${SramAxsConstants.SERVICE_UUID_RELEVANT}'),
);
final characteristic = service.characteristics.firstWhere(
(e) => e.uuid.toLowerCase() == SramAxsConstants.TRIGGER_UUID.toLowerCase(),
orElse: () => throw Exception('Characteristic not found: ${SramAxsConstants.TRIGGER_UUID}'),
);
await UniversalBle.subscribeNotifications(device.deviceId, service.uuid, characteristic.uuid);
// add both buttons
_singleClickButton();
_doubleClickButton();
}
ControllerButton _singleClickButton() => getOrAddButton(
'SRAM Tap',
() => const ControllerButton('SRAM Tap', action: InGameAction.shiftUp),
);
ControllerButton _doubleClickButton() => getOrAddButton(
'SRAM Double Tap',
() => const ControllerButton('SRAM Double Tap', action: InGameAction.shiftDown),
);
void _emitClick(ControllerButton button) {
// Use the common pipeline so long-press handling and app action execution stays consistent.
handleButtonsClickedWithoutLongPressSupport([button]);
}
void _registerTap() {
final windowMs = core.settings.getSramAxsDoubleClickWindowMs();
_tapCount++;
// First tap: start a timer. If no second tap arrives in time => single click.
if (_tapCount == 1) {
_singleClickTimer?.cancel();
_singleClickTimer = Timer(Duration(milliseconds: windowMs), () {
if (_tapCount == 1) {
_emitClick(_singleClickButton());
}
_tapCount = 0;
});
return;
}
// Second tap within window: double click.
if (_tapCount == 2) {
_singleClickTimer?.cancel();
_singleClickTimer = null;
_emitClick(_doubleClickButton());
_tapCount = 0;
return;
}
// If we get more than two taps fast, treat as a double click and restart counting.
_singleClickTimer?.cancel();
_singleClickTimer = null;
_emitClick(_doubleClickButton());
_tapCount = 0;
}
@override
Future<void> processCharacteristic(String characteristic, Uint8List bytes) async {
if (kDebugMode) {
debugPrint('SramAxs: Received data on characteristic $characteristic: ${bytesToHex(bytes)}');
}
if (characteristic.toLowerCase() == SramAxsConstants.TRIGGER_UUID.toLowerCase()) {
// At the moment we can only detect "some button pressed". We therefore interpret each
// notification as a tap and provide two logical buttons (single & double click).
_registerTap();
}
return Future.value();
}
@override
Widget showInformation(BuildContext context) {
final windowMs = core.settings.getSramAxsDoubleClickWindowMs();
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
spacing: 12,
children: [
super.showInformation(context),
Text(
"Unfortunately, at the moment it's not possible to determine which physical button was pressed on your SRAM AXS device. Let us know if you have a contact at SRAM who can help :)\n\n"
'So the app exposes two logical buttons:\n'
'• SRAM Tap, assigned to Shift Up\n'
'• SRAM Double Tap, assigned to Shift Down\n\n'
'You can assign an action to each in the app settings.',
).xSmall,
Builder(
builder: (context) {
return PrimaryButton(
size: ButtonSize.small,
trailing: Container(
padding: const EdgeInsets.symmetric(horizontal: 4),
decoration: BoxDecoration(
color: Theme.of(context).colorScheme.primary,
borderRadius: BorderRadius.circular(4),
),
child: Text('${windowMs}ms'),
),
onPressed: () {
final values = [
for (var v = 150; v <= 600; v += 50) v,
];
showDropdown(
context: context,
builder: (b) => DropdownMenu(
children: values
.map(
(v) => MenuButton(
child: Text('${v}ms'),
onPressed: (c) async {
await core.settings.setSramAxsDoubleClickWindowMs(v);
// Force rebuild to show new value.
(context as Element).markNeedsBuild();
},
),
)
.toList(),
),
);
},
child: const Text('Double-click window:'),
);
},
),
],
);
}
}
class SramAxsConstants {
static const String SERVICE_UUID = "0000fe51-0000-1000-8000-00805f9b34fb";
static const String SERVICE_UUID_RELEVANT = "d9050053-90aa-4c7c-b036-1e01fb8eb7ee";
static const String TRIGGER_UUID = "d9050054-90aa-4c7c-b036-1e01fb8eb7ee";
}

View File

@@ -0,0 +1,16 @@
import 'package:bike_control/utils/actions/base_actions.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:bike_control/utils/keymap/keymap.dart';
import 'package:flutter/foundation.dart';
abstract class TrainerConnection {
final String title;
List<InGameAction> supportedActions;
final ValueNotifier<bool> isStarted = ValueNotifier(false);
final ValueNotifier<bool> isConnected = ValueNotifier(false);
TrainerConnection({required this.title, required this.supportedActions});
Future<ActionResult> sendAction(KeyPair keyPair, {required bool isKeyDown, required bool isKeyUp});
}

View File

@@ -0,0 +1,10 @@
import 'package:bike_control/bluetooth/devices/zwift/zwift_ride.dart';
import '../zwift/constants.dart';
class WahooKickrBikePro extends ZwiftRide {
WahooKickrBikePro(super.scanResult) : super();
@override
String get customServiceId => ZwiftConstants.ZWIFT_CUSTOM_SERVICE_UUID;
}

View File

@@ -2,7 +2,7 @@ import 'dart:collection';
import 'dart:typed_data';
import 'package:flutter/material.dart';
import 'package:swift_control/utils/keymap/buttons.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:universal_ble/universal_ble.dart';
import '../bluetooth_device.dart';

View File

@@ -0,0 +1,151 @@
import 'dart:typed_data';
import 'package:bike_control/bluetooth/devices/zwift/ftms_mdns_emulator.dart';
import 'package:bike_control/bluetooth/messages/notification.dart';
import 'package:bike_control/utils/actions/base_actions.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:bike_control/utils/keymap/keymap.dart';
import 'package:universal_ble/universal_ble.dart';
import '../bluetooth_device.dart';
class WahooKickrHeadwind extends BluetoothDevice {
// Current mode state
HeadwindMode _currentMode = HeadwindMode.unknown;
int _currentSpeed = 0;
WahooKickrHeadwind(super.scanResult)
: super(
availableButtons: const [],
isBeta: true,
);
@override
Future<void> handleServices(List<BleService> services) async {
final service = services.firstWhere(
(e) => e.uuid == WahooKickrHeadwindConstants.SERVICE_UUID.toLowerCase(),
orElse: () => throw Exception('Service not found: ${WahooKickrHeadwindConstants.SERVICE_UUID}'),
);
final characteristic = service.characteristics.firstWhere(
(e) => e.uuid == WahooKickrHeadwindConstants.CHARACTERISTIC_UUID.toLowerCase(),
orElse: () => throw Exception('Characteristic not found: ${WahooKickrHeadwindConstants.CHARACTERISTIC_UUID}'),
);
// Subscribe to notifications for status updates
await UniversalBle.subscribeNotifications(device.deviceId, service.uuid, characteristic.uuid);
}
@override
Future<void> processCharacteristic(String characteristic, Uint8List bytes) {
// Analyze the received bytes to determine current state
actionStreamInternal.add(LogNotification('Received ${bytesToHex(bytes)} from Headwind $characteristic'));
if (bytes.length >= 4 && bytes[0] == 0xFD && bytes[1] == 0x01) {
final mode = bytes[3];
final speed = bytes[2];
switch (mode) {
case 0x02:
_currentMode = HeadwindMode.heartRate;
break;
case 0x03:
_currentMode = HeadwindMode.speed;
break;
case 0x01:
_currentMode = HeadwindMode.off;
break;
case 0x04:
_currentMode = HeadwindMode.manual;
_currentSpeed = speed;
break;
}
}
return Future.value();
}
Future<void> setSpeed(int speedPercent) async {
// Validate against the allowed speed values
const allowedSpeeds = [0, 25, 50, 75, 100];
if (!allowedSpeeds.contains(speedPercent)) {
throw ArgumentError('Speed must be one of: ${allowedSpeeds.join(", ")}');
}
final service = WahooKickrHeadwindConstants.SERVICE_UUID.toLowerCase();
final characteristic = WahooKickrHeadwindConstants.CHARACTERISTIC_UUID.toLowerCase();
// Check if manual mode is enabled, if not enable it first
if (_currentMode != HeadwindMode.manual) {
final manualModeData = Uint8List.fromList([0x04, 0x04]);
await UniversalBle.write(
device.deviceId,
service,
characteristic,
manualModeData,
);
_currentMode = HeadwindMode.manual;
}
// Command format: [0x02, speed_value]
// Speed value: 0x00 to 0x64 (0-100 in hex)
final data = Uint8List.fromList([0x02, speedPercent]);
await UniversalBle.write(
device.deviceId,
service,
characteristic,
data,
);
_currentSpeed = speedPercent;
}
Future<void> setHeartRateMode() async {
final service = WahooKickrHeadwindConstants.SERVICE_UUID.toLowerCase();
final characteristic = WahooKickrHeadwindConstants.CHARACTERISTIC_UUID.toLowerCase();
// Command format: [0x04, 0x02] for HR mode
final data = Uint8List.fromList([0x04, 0x02]);
await UniversalBle.write(
device.deviceId,
service,
characteristic,
data,
);
_currentMode = HeadwindMode.heartRate;
}
Future<ActionResult> handleKeypair(KeyPair keyPair, {required bool isKeyDown}) async {
if (!isKeyDown) {
return NotHandled('');
}
try {
if (keyPair.inGameAction == InGameAction.headwindSpeed) {
final speed = keyPair.inGameActionValue ?? 0;
await setSpeed(speed);
return Success('Headwind speed set to $speed%');
} else if (keyPair.inGameAction == InGameAction.headwindHeartRateMode) {
await setHeartRateMode();
return Success('Headwind set to Heart Rate mode');
}
} catch (e) {
return Error('Failed to control Headwind: $e');
}
return NotHandled('');
}
}
class WahooKickrHeadwindConstants {
// Wahoo KICKR Headwind service and characteristic UUIDs
// These are standard Wahoo fitness equipment UUIDs
static const String SERVICE_UUID = "A026EE0C-0A7D-4AB3-97FA-F1500F9FEB8B";
static const String CHARACTERISTIC_UUID = "A026E038-0A7D-4AB3-97FA-F1500F9FEB8B";
}
enum HeadwindMode {
unknown,
heartRate, // HR mode (0x02)
speed, // Speed mode (0x03)
off, // OFF mode (0x01)
manual, // Manual speed mode (0x04)
}

View File

@@ -1,7 +1,7 @@
import 'dart:typed_data';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:flutter/material.dart';
import 'package:swift_control/utils/keymap/buttons.dart';
class ZwiftConstants {
static const ZWIFT_CUSTOM_SERVICE_UUID = "00000001-19CA-4651-86E5-FA29DCDD09D1";
@@ -71,13 +71,13 @@ class ZwiftButtons {
);
static const ControllerButton navigationLeft = ControllerButton(
'navigationLeft',
action: InGameAction.navigateLeft,
action: InGameAction.steerLeft,
icon: Icons.keyboard_arrow_left,
color: Colors.black,
);
static const ControllerButton navigationRight = ControllerButton(
'navigationRight',
action: InGameAction.navigateRight,
action: InGameAction.steerRight,
icon: Icons.keyboard_arrow_right,
color: Colors.black,
);
@@ -99,9 +99,13 @@ class ZwiftButtons {
static const ControllerButton powerUpLeft = ControllerButton('powerUpLeft', action: InGameAction.shiftDown);
// right controller
static const ControllerButton a = ControllerButton('a', action: null, color: Colors.lightGreen);
static const ControllerButton b = ControllerButton('b', action: null, color: Colors.pinkAccent);
static const ControllerButton z = ControllerButton('z', action: null, color: Colors.deepOrangeAccent);
static const ControllerButton a = ControllerButton('a', action: InGameAction.select, color: Colors.lightGreen);
static const ControllerButton b = ControllerButton('b', action: InGameAction.back, color: Colors.pinkAccent);
static const ControllerButton z = ControllerButton(
'z',
action: InGameAction.rideOnBomb,
color: Colors.deepOrangeAccent,
);
static const ControllerButton y = ControllerButton('y', action: null, color: Colors.lightBlue);
static const ControllerButton onOffRight = ControllerButton('onOffRight', action: InGameAction.toggleUi);
static const ControllerButton sideButtonRight = ControllerButton('sideButtonRight', action: InGameAction.shiftUp);

View File

@@ -0,0 +1,503 @@
import 'dart:io';
import 'package:bike_control/bluetooth/devices/trainer_connection.dart';
import 'package:bike_control/bluetooth/devices/zwift/constants.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zp.pbenum.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zwift.pb.dart' show RideKeyPadStatus;
import 'package:bike_control/bluetooth/devices/zwift/zwift_ride.dart';
import 'package:bike_control/bluetooth/messages/notification.dart';
import 'package:bike_control/gen/l10n.dart';
import 'package:bike_control/utils/actions/base_actions.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:bike_control/utils/keymap/keymap.dart';
import 'package:dartx/dartx.dart';
import 'package:flutter/foundation.dart';
import 'package:nsd/nsd.dart';
class FtmsMdnsEmulator extends TrainerConnection {
ServerSocket? _tcpServer;
Registration? _mdnsRegistration;
static const String connectionTitle = 'Zwift Network Emulator';
Socket? _socket;
var lastMessageId = 0;
FtmsMdnsEmulator()
: super(
title: connectionTitle,
supportedActions: [
InGameAction.shiftUp,
InGameAction.shiftDown,
InGameAction.uturn,
InGameAction.steerLeft,
InGameAction.steerRight,
InGameAction.openActionBar,
InGameAction.usePowerUp,
InGameAction.select,
InGameAction.back,
InGameAction.rideOnBomb,
],
);
Future<void> startServer() async {
isStarted.value = true;
print('Starting mDNS server...');
// Get local IP
final interfaces = await NetworkInterface.list();
InternetAddress? localIP;
for (final interface in interfaces) {
for (final addr in interface.addresses) {
if (addr.type == InternetAddressType.IPv4 && !addr.isLoopback) {
localIP = addr;
break;
}
}
if (localIP != null) break;
}
if (localIP == null) {
throw 'Could not find network interface';
}
_createTcpServer();
if (kDebugMode) {
enableLogging(LogTopic.calls);
enableLogging(LogTopic.errors);
}
disableServiceTypeValidation(true);
_mdnsRegistration = await register(
Service(
name: 'KICKR BIKE PRO 1337',
addresses: [localIP],
port: 36867,
type: '_wahoo-fitness-tnp._tcp',
txt: {
'ble-service-uuids': Uint8List.fromList('FC82'.codeUnits),
'mac-address': Uint8List.fromList('50-50-25-6C-66-9C'.codeUnits),
'serial-number': Uint8List.fromList('244700181'.codeUnits),
},
),
);
print('Server started - advertising service!');
}
void stop() {
isStarted.value = false;
isConnected.value = false;
_tcpServer?.close();
if (_mdnsRegistration != null) {
unregister(_mdnsRegistration!);
}
_tcpServer = null;
_mdnsRegistration = null;
_socket = null;
print('Stopped FtmsMdnsEmulator');
}
Future<void> _createTcpServer() async {
try {
_tcpServer = await ServerSocket.bind(
InternetAddress.anyIPv6,
36867,
shared: true,
v6Only: false,
);
} catch (e) {
if (kDebugMode) {
print('Failed to start server: $e');
}
rethrow;
}
if (kDebugMode) {
print('Server started on port ${_tcpServer!.port}');
}
// Accept connection
_tcpServer!.listen(
(Socket socket) {
_socket = socket;
isConnected.value = true;
if (kDebugMode) {
print('Client connected: ${socket.remoteAddress.address}:${socket.remotePort}');
}
core.connection.signalNotification(
AlertNotification(LogLevel.LOGLEVEL_INFO, AppLocalizations.current.connected),
);
// Listen for data from the client
socket.listen(
(List<int> data) {
if (kDebugMode) {
print('Received message: ${bytesToHex(data)}');
}
final mutable = data.toList();
while (mutable.isNotEmpty) {
final msgVersion = mutable.takeUInt8();
final msgId = mutable.takeUInt8();
lastMessageId = msgId;
final seqNum = mutable.takeUInt8();
final respCode = mutable.takeUInt8(); // Response Code
final length = mutable.takeUInt16BE(); // Length of the message body
final body = mutable.takeBytes(length);
if (kDebugMode) {
print('Parsed message: ID: $msgId, Body: ${bytesToHex(body)}');
}
Uint8List buildHeader(int responseCode, int bodyLength) {
return Uint8List.fromList([
msgVersion,
msgId,
seqNum,
responseCode,
(bodyLength >> 8) & 0xFF,
bodyLength & 0xFF,
]);
}
switch (msgId) {
case FtmsMdnsConstants.DC_MESSAGE_DISCOVER_SERVICES:
final body = hexToBytes(ZwiftConstants.ZWIFT_RIDE_CUSTOM_SERVICE_UUID.toNonDash());
final header = buildHeader(FtmsMdnsConstants.DC_RC_REQUEST_COMPLETED_SUCCESSFULLY, body.length);
final bytes = [...header, ...body];
// Expected 0101000000100000fc8200001000800000805f9b34fb
// Got 0101000000100000fc8200001000800000805f9b34fb
_write(socket, bytes);
case FtmsMdnsConstants.DC_MESSAGE_DISCOVER_CHARACTERISTICS:
final rawUUID = body.takeBytes(16);
final serviceUUID = bytesToHex(rawUUID).toUUID();
if (serviceUUID == ZwiftConstants.ZWIFT_RIDE_CUSTOM_SERVICE_UUID) {
final responseBody = [
...rawUUID,
...hexToBytes(ZwiftConstants.ZWIFT_SYNC_RX_CHARACTERISTIC_UUID.toNonDash()),
...[
_propertyVal(['write']),
],
...hexToBytes(ZwiftConstants.ZWIFT_ASYNC_CHARACTERISTIC_UUID.toNonDash()),
...[
_propertyVal(['notify']),
],
...hexToBytes(ZwiftConstants.ZWIFT_SYNC_TX_CHARACTERISTIC_UUID.toNonDash()),
...[
_propertyVal(['notify']),
],
];
final responseData = [
...buildHeader(
FtmsMdnsConstants.DC_RC_REQUEST_COMPLETED_SUCCESSFULLY,
responseBody.length,
),
...responseBody,
];
// OK: 0102010000430000fc8200001000800000805f9b34fb0000000319ca465186e5fa29dcdd09d1020000000219ca465186e5fa29dcdd09d1040000000419ca465186e5fa29dcdd09d104
_write(socket, responseData);
}
case FtmsMdnsConstants.DC_MESSAGE_READ_CHARACTERISTIC:
final rawUUID = body.takeBytes(16);
final characteristicUUID = bytesToHex(rawUUID).toUUID();
print(
'Got Read Characteristic UUID: $characteristicUUID',
);
final responseBody = rawUUID;
final responseData = [
...buildHeader(
FtmsMdnsConstants.DC_RC_REQUEST_COMPLETED_SUCCESSFULLY,
responseBody.length,
),
...responseBody,
];
_write(socket, responseData);
case FtmsMdnsConstants.DC_MESSAGE_WRITE_CHARACTERISTIC:
final rawUUID = body.takeBytes(16);
final characteristicUUID = bytesToHex(rawUUID).toUUID();
final characteristicData = body.takeBytes(body.length);
print(
'Got Write Characteristic UUID: $characteristicUUID, Data: ${bytesToHex(characteristicData)}',
);
final responseBody = rawUUID;
final responseData = [
...buildHeader(
FtmsMdnsConstants.DC_RC_REQUEST_COMPLETED_SUCCESSFULLY,
responseBody.length,
),
...responseBody,
];
_write(socket, responseData);
final response = core.zwiftEmulator.handleWriteRequest(
characteristicUUID,
Uint8List.fromList(characteristicData),
);
if (response != null) {
final seqNum = (lastMessageId + 1) % 256;
lastMessageId = seqNum;
final responseBody = [
...hexToBytes(ZwiftConstants.ZWIFT_SYNC_TX_CHARACTERISTIC_UUID.toLowerCase().toNonDash()),
...response,
];
final responseData = [
// header
...Uint8List.fromList([
msgVersion,
FtmsMdnsConstants.DC_MESSAGE_CHARACTERISTIC_NOTIFICATION,
seqNum,
FtmsMdnsConstants.DC_RC_REQUEST_COMPLETED_SUCCESSFULLY,
(responseBody.length >> 8) & 0xFF,
responseBody.length & 0xFF,
]),
// body
...responseBody,
];
// 0106050000180000000419ca465186e5fa29dcdd09d1526964654f6e0203
_write(socket, responseData);
if (response.contentEquals(ZwiftConstants.RIDE_ON)) {
_sendKeepAlive();
}
}
return;
case FtmsMdnsConstants.DC_MESSAGE_ENABLE_CHARACTERISTIC_NOTIFICATIONS:
final rawUUID = body.takeBytes(16);
final characteristicUUID = bytesToHex(rawUUID).toUUID();
final enabled = body.takeUInt8();
print(
'Got Enable Notifications for Characteristic UUID: $characteristicUUID, Enabled: $enabled',
);
final responseBody = rawUUID;
final responseData = [
...buildHeader(
FtmsMdnsConstants.DC_RC_REQUEST_COMPLETED_SUCCESSFULLY,
responseBody.length,
),
...responseBody,
];
_write(socket, responseData);
case FtmsMdnsConstants.DC_MESSAGE_CHARACTERISTIC_NOTIFICATION:
print('Hamlo');
default:
throw 'DC_ERROR_UNKNOWN_MESSAGE_TYPE';
}
}
},
onDone: () {
print('Client disconnected: $socket');
isConnected.value = false;
core.connection.signalNotification(
AlertNotification(LogLevel.LOGLEVEL_INFO, AppLocalizations.current.disconnected),
);
_socket = null;
},
);
},
);
}
void _write(Socket socket, List<int> responseData) {
if (kDebugMode) {
print('Sending response: ${bytesToHex(responseData)}');
}
socket.add(responseData);
}
int _propertyVal(List<String> properties) {
int res = 0;
if (properties.contains('read')) res |= 0x01;
if (properties.contains('write')) res |= 0x02;
if (properties.contains('indicate')) res |= 0x03;
if (properties.contains('notify')) res |= 0x04;
return res;
}
@override
Future<ActionResult> sendAction(KeyPair keyPair, {required bool isKeyDown, required bool isKeyUp}) async {
final button = switch (keyPair.inGameAction) {
InGameAction.shiftUp => RideButtonMask.SHFT_UP_R_BTN,
InGameAction.shiftDown => RideButtonMask.SHFT_UP_L_BTN,
InGameAction.uturn => RideButtonMask.DOWN_BTN,
InGameAction.steerLeft => RideButtonMask.LEFT_BTN,
InGameAction.steerRight => RideButtonMask.RIGHT_BTN,
InGameAction.openActionBar => RideButtonMask.UP_BTN,
InGameAction.usePowerUp => RideButtonMask.Y_BTN,
InGameAction.select => RideButtonMask.A_BTN,
InGameAction.back => RideButtonMask.B_BTN,
InGameAction.rideOnBomb => RideButtonMask.Z_BTN,
_ => null,
};
if (button == null) {
return NotHandled('Action ${keyPair.inGameAction!.name} not supported by Zwift Emulator');
}
if (isKeyDown) {
final status = RideKeyPadStatus()
..buttonMap = (~button.mask) & 0xFFFFFFFF
..analogPaddles.clear();
final bytes = status.writeToBuffer();
final commandProto = _buildNotify(
ZwiftConstants.ZWIFT_ASYNC_CHARACTERISTIC_UUID,
Uint8List.fromList([
Opcode.CONTROLLER_NOTIFICATION.value,
...bytes,
]),
);
_write(_socket!, commandProto);
}
if (isKeyUp) {
final zero = _buildNotify(
ZwiftConstants.ZWIFT_ASYNC_CHARACTERISTIC_UUID,
Uint8List.fromList([Opcode.CONTROLLER_NOTIFICATION.value, 0x08, 0xFF, 0xFF, 0xFF, 0xFF, 0x0F]),
);
_write(_socket!, zero);
}
if (kDebugMode) {
print('Sent action $isKeyUp vs $isKeyDown ${keyPair.inGameAction!.title} to Zwift Emulator');
}
return Success('Sent action: ${keyPair.inGameAction!.title}');
}
List<int> _buildNotify(String uuid, final List<int> data) {
final seqNum = (lastMessageId + 1) % 256;
lastMessageId = seqNum;
final responseBody = [
...hexToBytes(uuid.toLowerCase().toNonDash()),
...data,
];
final responseData = [
// header
...Uint8List.fromList([
0x01,
FtmsMdnsConstants.DC_MESSAGE_CHARACTERISTIC_NOTIFICATION,
seqNum,
FtmsMdnsConstants.DC_RC_REQUEST_COMPLETED_SUCCESSFULLY,
(responseBody.length >> 8) & 0xFF,
responseBody.length & 0xFF,
]),
// body
...responseBody,
];
return responseData;
}
Future<void> _sendKeepAlive() async {
await Future.delayed(const Duration(seconds: 5));
if (_socket != null) {
_write(
_socket!,
_buildNotify(
ZwiftConstants.ZWIFT_SYNC_TX_CHARACTERISTIC_UUID,
hexToBytes('B70100002041201C00180004001B4F00B701000020798EC5BDEFCBE4563418269E4926FBE1'),
),
);
_sendKeepAlive();
}
}
}
extension on String {
String toNonDash() {
return replaceAll('-', '');
}
String toUUID() {
return '${substring(0, 8)}-${substring(8, 12)}-${substring(12, 16)}-${substring(16, 20)}-${substring(20)}';
}
}
extension on List<int> {
int takeUInt8() {
final value = this[0];
removeAt(0);
return value;
}
int readUInt8(int offset) {
return this[offset];
}
int takeUInt16BE() {
final value = (this[0] << 8) | this[0 + 1];
removeAt(0);
removeAt(0);
return value;
}
List<int> takeBytes(int length) {
final value = sublist(0, length);
removeRange(0, length);
return value;
}
int readUInt16BE(int i) {
final value = (this[i] << 8) | this[i + 1];
return value;
}
}
String bytesToHex(List<int> bytes, {bool spaced = false}) {
return bytes.map((byte) => byte.toRadixString(16).padLeft(2, '0')).join(spaced ? ' ' : '');
}
String bytesToReadableHex(List<int> bytes) {
return bytes.map((byte) => byte.toRadixString(16).padLeft(2, '0')).join(' ');
}
List<int> hexToBytes(String hex) {
final bytes = <int>[];
for (var i = 0; i < hex.length; i += 2) {
final byte = hex.substring(i, i + 2);
bytes.add(int.parse(byte, radix: 16));
}
return bytes;
}
class FtmsMdnsConstants {
static const DC_RC_REQUEST_COMPLETED_SUCCESSFULLY = 0; // Request completed successfully
static const DC_RC_UNKNOWN_MESSAGE_TYPE = 1; // Unknown Message Type
static const DC_RC_UNEXPECTED_ERROR = 2; // Unexpected Error
static const DC_RC_SERVICE_NOT_FOUND = 3; // Service Not Found
static const DC_RC_CHARACTERISTIC_NOT_FOUND = 4; // Characteristic Not Found
static const DC_RC_CHARACTERISTIC_OPERATION_NOT_SUPPORTED =
5; // Characteristic Operation Not Supported (See Characteristic Properties)
static const DC_RC_CHARACTERISTIC_WRITE_FAILED_INVALID_SIZE =
6; // Characteristic Write Failed Invalid characteristic data size
static const DC_RC_UNKNOWN_PROTOCOL_VERSION =
7; // Unknown Protocol Version the command contains a protocol version that the device does not recognize
static const DC_MESSAGE_DISCOVER_SERVICES = 0x01; // Discover Services
static const DC_MESSAGE_DISCOVER_CHARACTERISTICS = 0x02; // Discover Characteristics
static const DC_MESSAGE_READ_CHARACTERISTIC = 0x03; // Read Characteristic
static const DC_MESSAGE_WRITE_CHARACTERISTIC = 0x04; // Write Characteristic
static const DC_MESSAGE_ENABLE_CHARACTERISTIC_NOTIFICATIONS = 0x05; // Enable Characteristic Notifications
static const DC_MESSAGE_CHARACTERISTIC_NOTIFICATION = 0x06; // Characteristic Notification
}

View File

@@ -1,7 +1,7 @@
import 'package:flutter/foundation.dart';
import 'package:swift_control/bluetooth/devices/zwift/protocol/zwift.pb.dart';
import 'package:swift_control/bluetooth/devices/zwift/zwift_device.dart';
import 'package:swift_control/utils/keymap/buttons.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zwift.pb.dart';
import 'package:bike_control/bluetooth/devices/zwift/zwift_device.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'constants.dart';

View File

@@ -1,10 +1,14 @@
import 'package:dartx/dartx.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:swift_control/bluetooth/devices/zwift/constants.dart';
import 'package:swift_control/bluetooth/devices/zwift/protocol/zp.pbenum.dart';
import 'package:swift_control/bluetooth/devices/zwift/zwift_ride.dart';
import 'package:swift_control/pages/markdown.dart';
import 'package:swift_control/widgets/warning.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
import 'package:bike_control/bluetooth/devices/zwift/constants.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zp.pbenum.dart';
import 'package:bike_control/bluetooth/devices/zwift/zwift_ride.dart';
import 'package:bike_control/gen/l10n.dart';
import 'package:bike_control/pages/markdown.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/i18n_extension.dart';
import 'package:bike_control/widgets/ui/warning.dart';
class ZwiftClickV2 extends ZwiftRide {
ZwiftClickV2(super.scanResult)
@@ -24,67 +28,114 @@ class ZwiftClickV2 extends ZwiftRide {
],
);
bool _noLongerSendsEvents = false;
@override
List<int> get startCommand => ZwiftConstants.RIDE_ON + ZwiftConstants.RESPONSE_START_CLICK_V2;
@override
String get latestFirmwareVersion => '1.1.0';
@override
bool get canVibrate => false;
@override
Future<void> setupHandshake() async {
super.setupHandshake();
await sendCommandBuffer(Uint8List.fromList([0xFF, 0x04, 0x00]));
}
@override
Future<void> processData(Uint8List bytes) {
if (bytes.startsWith(ZwiftConstants.RESPONSE_STOPPED_CLICK_V2_VARIANT_1) ||
bytes.startsWith(ZwiftConstants.RESPONSE_STOPPED_CLICK_V2_VARIANT_2)) {
_noLongerSendsEvents = true;
}
return super.processData(bytes);
}
@override
Widget showInformation(BuildContext context) {
return Column(
mainAxisAlignment: MainAxisAlignment.start,
children: [
super.showInformation(context),
return StatefulBuilder(
builder: (context, setState) {
return Column(
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
spacing: 8,
children: [
super.showInformation(context),
if (isConnected)
Warning(
children: [
Text(
'''To make your Zwift Click V2 work best you should connect it in the Zwift app once each day.\nIf you don't do that SwiftControl will need to reconnect every minute.
1. Open Zwift app
2. Log in (subscription not required) and open the device connection screen
3. Connect your Trainer, then connect the Zwift Click V2
4. Close the Zwift app again and connect again in SwiftControl''',
),
Row(
children: [
TextButton(
onPressed: () {
sendCommand(Opcode.RESET, null);
},
child: Text('Reset now'),
),
TextButton(
onPressed: () {
Navigator.push(
context,
MaterialPageRoute(
builder: (_) => MarkdownPage(assetPath: 'TROUBLESHOOTING.md'),
if (isConnected && _noLongerSendsEvents)
if (core.settings.getShowZwiftClickV2ReconnectWarning())
Warning(
children: [
Row(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Expanded(
child: Text(
AppLocalizations.of(context).clickV2Instructions,
).xSmall,
),
);
},
child: Text('Troubleshooting'),
),
if (kDebugMode)
TextButton(
onPressed: () {
test();
},
child: Text('Test'),
IconButton.link(
icon: Icon(Icons.close),
onPressed: () {
core.settings.setShowZwiftClickV2ReconnectWarning(false);
setState(() {});
},
),
],
),
],
),
],
),
],
Column(
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
spacing: 8,
children: [
GhostButton(
onPressed: () {
sendCommand(Opcode.RESET, null);
},
child: Text('Reset now'),
),
OutlineButton(
onPressed: () {
Navigator.push(
context,
MaterialPageRoute(
builder: (_) => MarkdownPage(assetPath: 'TROUBLESHOOTING.md'),
),
);
},
leading: const Icon(Icons.open_in_new),
child: Text(context.i18n.troubleshootingGuide),
),
],
),
],
)
else
Warning(
important: false,
children: [
Text(
AppLocalizations.of(context).clickV2EventInfo,
).xSmall,
LinkButton(
child: Text(context.i18n.troubleshootingGuide),
onPressed: () {
Navigator.push(
context,
MaterialPageRoute(
builder: (_) => MarkdownPage(assetPath: 'TROUBLESHOOTING.md'),
),
);
},
),
],
),
],
);
},
);
}

View File

@@ -1,13 +1,14 @@
import 'dart:async';
import 'package:bike_control/bluetooth/devices/bluetooth_device.dart';
import 'package:bike_control/bluetooth/devices/zwift/constants.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zp.pbenum.dart';
import 'package:bike_control/bluetooth/messages/notification.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:bike_control/utils/single_line_exception.dart';
import 'package:dartx/dartx.dart';
import 'package:flutter/foundation.dart';
import 'package:swift_control/bluetooth/devices/bluetooth_device.dart';
import 'package:swift_control/bluetooth/devices/zwift/constants.dart';
import 'package:swift_control/bluetooth/messages/notification.dart';
import 'package:swift_control/main.dart';
import 'package:swift_control/utils/keymap/buttons.dart';
import 'package:swift_control/utils/single_line_exception.dart';
import 'package:universal_ble/universal_ble.dart';
abstract class ZwiftDevice extends BluetoothDevice {
@@ -17,27 +18,41 @@ abstract class ZwiftDevice extends BluetoothDevice {
List<ControllerButton>? _lastButtonsClicked;
BleService? customService;
String get latestFirmwareVersion;
List<int> get startCommand => ZwiftConstants.RIDE_ON + ZwiftConstants.RESPONSE_START_CLICK;
String get customServiceId => ZwiftConstants.ZWIFT_CUSTOM_SERVICE_UUID;
bool get canVibrate => false;
@override
Future<void> handleServices(List<BleService> services) async {
final customService = services.firstOrNullWhere((service) => service.uuid == customServiceId.toLowerCase());
customService =
services.firstOrNullWhere(
(service) => service.uuid == ZwiftConstants.ZWIFT_RIDE_CUSTOM_SERVICE_UUID.toLowerCase(),
) ??
services.firstOrNullWhere(
(service) => service.uuid == ZwiftConstants.ZWIFT_CUSTOM_SERVICE_UUID.toLowerCase(),
);
if (customService == null) {
actionStreamInternal.add(
AlertNotification(
LogLevel.LOGLEVEL_ERROR,
'You may need to update the firmware of ${scanResult.name} in Zwift Companion app',
),
);
throw Exception(
'Custom service $customServiceId not found for device $this ${device.name ?? device.rawName}.\nYou may need to update the firmware in Zwift Companion app.\nWe found: ${services.joinToString(transform: (s) => s.uuid)}',
'Custom service ${[ZwiftConstants.ZWIFT_RIDE_CUSTOM_SERVICE_UUID, ZwiftConstants.ZWIFT_RIDE_CUSTOM_SERVICE_UUID]} not found for device $this ${device.name ?? device.rawName}.\nYou may need to update the firmware in Zwift Companion app.\nWe found: ${services.joinToString(transform: (s) => s.uuid)}',
);
}
final asyncCharacteristic = customService.characteristics.firstOrNullWhere(
final asyncCharacteristic = customService!.characteristics.firstOrNullWhere(
(characteristic) => characteristic.uuid == ZwiftConstants.ZWIFT_ASYNC_CHARACTERISTIC_UUID.toLowerCase(),
);
final syncTxCharacteristic = customService.characteristics.firstOrNullWhere(
final syncTxCharacteristic = customService!.characteristics.firstOrNullWhere(
(characteristic) => characteristic.uuid == ZwiftConstants.ZWIFT_SYNC_TX_CHARACTERISTIC_UUID.toLowerCase(),
);
syncRxCharacteristic = customService.characteristics.firstOrNullWhere(
syncRxCharacteristic = customService!.characteristics.firstOrNullWhere(
(characteristic) => characteristic.uuid == ZwiftConstants.ZWIFT_SYNC_RX_CHARACTERISTIC_UUID.toLowerCase(),
);
@@ -45,14 +60,15 @@ abstract class ZwiftDevice extends BluetoothDevice {
throw Exception('Characteristics not found');
}
await UniversalBle.subscribeNotifications(device.deviceId, customService.uuid, asyncCharacteristic.uuid);
await UniversalBle.subscribeIndications(device.deviceId, customService.uuid, syncTxCharacteristic.uuid);
await UniversalBle.subscribeNotifications(device.deviceId, customService!.uuid, asyncCharacteristic.uuid);
await UniversalBle.subscribeIndications(device.deviceId, customService!.uuid, syncTxCharacteristic.uuid);
await setupHandshake();
if (firmwareVersion != latestFirmwareVersion) {
if (firmwareVersion != latestFirmwareVersion && firmwareVersion != null) {
actionStreamInternal.add(
LogNotification(
AlertNotification(
LogLevel.LOGLEVEL_WARNING,
'A new firmware version is available for ${device.name ?? device.rawName}: $latestFirmwareVersion (current: $firmwareVersion). Please update it in Zwift Companion app.',
),
);
@@ -62,7 +78,7 @@ abstract class ZwiftDevice extends BluetoothDevice {
Future<void> setupHandshake() async {
await UniversalBle.write(
device.deviceId,
customServiceId,
customService!.uuid,
syncRxCharacteristic!.uuid,
ZwiftConstants.RIDE_ON,
withoutResponse: true,
@@ -72,8 +88,10 @@ abstract class ZwiftDevice extends BluetoothDevice {
@override
Future<void> processCharacteristic(String characteristic, Uint8List bytes) async {
if (kDebugMode) {
print(
"${DateTime.now().toString().split(" ").last} Received data on $characteristic: ${bytes.map((e) => e.toRadixString(16).padLeft(2, '0')).join(' ')}",
actionStreamInternal.add(
LogNotification(
"${DateTime.now().toString().split(" ").last} Received data on $characteristic: ${bytes.map((e) => e.toRadixString(16).padLeft(2, '0')).join(' ')}",
),
);
}
if (bytes.isEmpty) {
@@ -117,7 +135,7 @@ abstract class ZwiftDevice extends BluetoothDevice {
case ZwiftConstants.BATTERY_LEVEL_TYPE:
if (batteryLevel != message[1]) {
batteryLevel = message[1];
connection.signalChange(this);
core.connection.signalChange(this);
}
break;
case ZwiftConstants.CLICK_NOTIFICATION_MESSAGE_TYPE:
@@ -134,10 +152,10 @@ abstract class ZwiftDevice extends BluetoothDevice {
}
@override
Future<void> handleButtonsClicked(List<ControllerButton>? buttonsClicked) async {
Future<void> handleButtonsClicked(List<ControllerButton>? buttonsClicked, {bool longPress = false}) async {
// the same messages are sent multiple times, so ignore
if (_lastButtonsClicked == null || _lastButtonsClicked?.contentEquals(buttonsClicked ?? []) == false) {
super.handleButtonsClicked(buttonsClicked);
super.handleButtonsClicked(buttonsClicked, longPress: longPress);
}
_lastButtonsClicked = buttonsClicked;
}
@@ -147,7 +165,7 @@ abstract class ZwiftDevice extends BluetoothDevice {
@override
Future<void> performDown(List<ControllerButton> buttonsClicked) async {
if (buttonsClicked.any(((e) => e.action == InGameAction.shiftDown || e.action == InGameAction.shiftUp)) &&
settings.getVibrationEnabled()) {
core.settings.getVibrationEnabled()) {
await _vibrate();
}
return super.performDown(buttonsClicked);
@@ -156,7 +174,8 @@ abstract class ZwiftDevice extends BluetoothDevice {
@override
Future<void> performClick(List<ControllerButton> buttonsClicked) async {
if (buttonsClicked.any(((e) => e.action == InGameAction.shiftDown || e.action == InGameAction.shiftUp)) &&
settings.getVibrationEnabled()) {
core.settings.getVibrationEnabled() &&
canVibrate) {
await _vibrate();
}
return super.performClick(buttonsClicked);
@@ -166,7 +185,7 @@ abstract class ZwiftDevice extends BluetoothDevice {
final vibrateCommand = Uint8List.fromList([...ZwiftConstants.VIBRATE_PATTERN, 0x20]);
await UniversalBle.write(
device.deviceId,
customServiceId,
customService!.uuid,
syncRxCharacteristic!.uuid,
vibrateCommand,
withoutResponse: true,

View File

@@ -1,96 +1,103 @@
import 'dart:io';
import 'package:bike_control/bluetooth/ble.dart';
import 'package:bike_control/bluetooth/devices/trainer_connection.dart';
import 'package:bike_control/bluetooth/devices/zwift/constants.dart';
import 'package:bike_control/bluetooth/devices/zwift/ftms_mdns_emulator.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zp.pb.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zwift.pbserver.dart' hide RideButtonMask;
import 'package:bike_control/bluetooth/devices/zwift/zwift_ride.dart';
import 'package:bike_control/bluetooth/messages/notification.dart';
import 'package:bike_control/gen/l10n.dart';
import 'package:bike_control/utils/actions/base_actions.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:bike_control/utils/keymap/keymap.dart';
import 'package:bike_control/utils/requirements/multi.dart';
import 'package:bike_control/widgets/title.dart';
import 'package:bluetooth_low_energy/bluetooth_low_energy.dart';
import 'package:dartx/dartx.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart' hide ConnectionState;
import 'package:permission_handler/permission_handler.dart';
import 'package:swift_control/bluetooth/ble.dart';
import 'package:swift_control/bluetooth/devices/zwift/constants.dart';
import 'package:swift_control/bluetooth/devices/zwift/protocol/zp.pbenum.dart';
import 'package:swift_control/bluetooth/devices/zwift/zwift_ride.dart';
import 'package:swift_control/main.dart';
import 'package:swift_control/utils/keymap/buttons.dart';
import 'package:swift_control/utils/requirements/multi.dart';
import 'package:swift_control/widgets/title.dart';
import 'protocol/zwift.pb.dart' show RideKeyPadStatus;
final zwiftEmulator = ZwiftEmulator();
class ZwiftEmulator {
static final List<InGameAction> supportedActions = [
InGameAction.shiftUp,
InGameAction.shiftDown,
InGameAction.uturn,
InGameAction.steerLeft,
InGameAction.steerRight,
InGameAction.openActionBar,
InGameAction.usePowerUp,
InGameAction.select,
InGameAction.back,
InGameAction.rideOnBomb,
];
ValueNotifier<bool> isConnected = ValueNotifier<bool>(false);
bool get isAdvertising => _isAdvertising;
class ZwiftEmulator extends TrainerConnection {
bool get isLoading => _isLoading;
final peripheralManager = PeripheralManager();
bool _isAdvertising = false;
static const String connectionTitle = 'Zwift BLE Emulator';
late final _peripheralManager = PeripheralManager();
bool _isLoading = false;
bool _isServiceAdded = false;
bool _isSubscribedToEvents = false;
Central? _central;
GATTCharacteristic? _asyncCharacteristic;
GATTCharacteristic? _syncTxCharacteristic;
ZwiftEmulator()
: super(
title: connectionTitle,
supportedActions: [
InGameAction.shiftUp,
InGameAction.shiftDown,
InGameAction.uturn,
InGameAction.steerLeft,
InGameAction.steerRight,
InGameAction.openActionBar,
InGameAction.usePowerUp,
InGameAction.select,
InGameAction.back,
InGameAction.rideOnBomb,
],
);
Future<void> reconnect() async {
await peripheralManager.stopAdvertising();
await peripheralManager.removeAllServices();
await _peripheralManager.stopAdvertising();
await _peripheralManager.removeAllServices();
_isServiceAdded = false;
_isAdvertising = false;
startAdvertising(() {});
}
Future<void> startAdvertising(VoidCallback onUpdate) async {
_isLoading = true;
isStarted.value = true;
onUpdate();
peripheralManager.stateChanged.forEach((state) {
_peripheralManager.stateChanged.forEach((state) {
print('Peripheral manager state: ${state.state}');
});
if (!kIsWeb && Platform.isAndroid) {
if (Platform.isAndroid) {
peripheralManager.connectionStateChanged.forEach((state) {
print('Peripheral connection state: ${state.state} of ${state.central.uuid}');
if (state.state == ConnectionState.connected) {
} else if (state.state == ConnectionState.disconnected) {
_central = null;
isConnected.value = false;
onUpdate();
}
});
}
_peripheralManager.connectionStateChanged.forEach((state) {
print('Peripheral connection state: ${state.state} of ${state.central.uuid}');
if (state.state == ConnectionState.connected) {
} else if (state.state == ConnectionState.disconnected) {
_central = null;
isConnected.value = false;
core.connection.signalNotification(
AlertNotification(LogLevel.LOGLEVEL_INFO, AppLocalizations.current.disconnected),
);
onUpdate();
}
});
final status = await Permission.bluetoothAdvertise.request();
if (!status.isGranted) {
print('Bluetooth advertise permission not granted');
_isAdvertising = false;
isStarted.value = false;
onUpdate();
return;
}
}
while (peripheralManager.state != BluetoothLowEnergyState.poweredOn) {
while (_peripheralManager.state != BluetoothLowEnergyState.poweredOn &&
core.settings.getZwiftBleEmulatorEnabled()) {
print('Waiting for peripheral manager to be powered on...');
if (settings.getLastTarget() == Target.thisDevice) {
if (core.settings.getLastTarget() == Target.thisDevice) {
return;
}
await Future.delayed(Duration(seconds: 1));
}
final syncTxCharacteristic = GATTCharacteristic.mutable(
_syncTxCharacteristic = GATTCharacteristic.mutable(
uuid: UUID.fromString(ZwiftConstants.ZWIFT_SYNC_TX_CHARACTERISTIC_UUID),
descriptors: [],
properties: [
@@ -116,7 +123,7 @@ class ZwiftEmulator {
if (!_isSubscribedToEvents) {
_isSubscribedToEvents = true;
peripheralManager.characteristicReadRequested.forEach((eventArgs) async {
_peripheralManager.characteristicReadRequested.forEach((eventArgs) async {
print('Read request for characteristic: ${eventArgs.characteristic.uuid}');
switch (eventArgs.characteristic.uuid.toString().toUpperCase()) {
@@ -124,7 +131,7 @@ class ZwiftEmulator {
print('Handling read request for SYNC TX characteristic');
break;
case BleUuid.DEVICE_INFORMATION_CHARACTERISTIC_BATTERY_LEVEL:
await peripheralManager.respondReadRequestWithValue(
await _peripheralManager.respondReadRequestWithValue(
eventArgs.request,
value: Uint8List.fromList([100]),
);
@@ -135,89 +142,78 @@ class ZwiftEmulator {
final request = eventArgs.request;
final trimmedValue = Uint8List.fromList([]);
await peripheralManager.respondReadRequestWithValue(
await _peripheralManager.respondReadRequestWithValue(
request,
value: trimmedValue,
);
// You can respond to read requests here if needed
});
peripheralManager.characteristicNotifyStateChanged.forEach((char) {
_peripheralManager.characteristicNotifyStateChanged.forEach((char) {
print(
'Notify state changed for characteristic: ${char.characteristic.uuid}: ${char.state}',
);
});
peripheralManager.characteristicWriteRequested.forEach((eventArgs) async {
_peripheralManager.characteristicWriteRequested.forEach((eventArgs) async {
_central = eventArgs.central;
isConnected.value = true;
final characteristic = eventArgs.characteristic;
final request = eventArgs.request;
final value = request.value;
print(
'Write request for characteristic: ${characteristic.uuid}',
core.connection.signalNotification(
AlertNotification(LogLevel.LOGLEVEL_INFO, AppLocalizations.current.connected),
);
switch (eventArgs.characteristic.uuid.toString().toUpperCase()) {
case ZwiftConstants.ZWIFT_SYNC_RX_CHARACTERISTIC_UUID:
print(
'Handling write request for SYNC RX characteristic, value: ${value.map((e) => e.toRadixString(16).padLeft(2, '0')).join(' ')}\n${String.fromCharCodes(value)}',
);
final handshake = [...ZwiftConstants.RIDE_ON, ...ZwiftConstants.RESPONSE_START_CLICK_V2];
final handshakeAlternative = ZwiftConstants.RIDE_ON; // e.g. Rouvy
if (value.contentEquals(handshake) || value.contentEquals(handshakeAlternative)) {
print('Sending handshake');
await peripheralManager.notifyCharacteristic(
_central!,
syncTxCharacteristic,
value: ZwiftConstants.RIDE_ON,
);
onUpdate();
}
break;
default:
print('Unhandled write request for characteristic: ${eventArgs.characteristic.uuid}');
final request = eventArgs.request;
final response = handleWriteRequest(eventArgs.characteristic.uuid.toString(), request.value);
if (response != null) {
await _peripheralManager.notifyCharacteristic(
_central!,
_syncTxCharacteristic!,
value: response,
);
onUpdate();
if (response == ZwiftConstants.RIDE_ON) {
_sendKeepAlive();
}
}
await peripheralManager.respondWriteRequest(request);
await _peripheralManager.respondWriteRequest(request);
});
}
// Device Information
await peripheralManager.addService(
GATTService(
uuid: UUID.fromString('180A'),
isPrimary: true,
characteristics: [
GATTCharacteristic.immutable(
uuid: UUID.fromString('2A29'),
value: Uint8List.fromList('SwiftControl'.codeUnits),
descriptors: [],
),
GATTCharacteristic.immutable(
uuid: UUID.fromString('2A25'),
value: Uint8List.fromList('09-B48123283828F1337'.codeUnits),
descriptors: [],
),
GATTCharacteristic.immutable(
uuid: UUID.fromString('2A27'),
value: Uint8List.fromList('A.0'.codeUnits),
descriptors: [],
),
GATTCharacteristic.immutable(
uuid: UUID.fromString('2A26'),
value: Uint8List.fromList((packageInfoValue?.version ?? '1.0.0').codeUnits),
descriptors: [],
),
],
includedServices: [],
),
);
if (!Platform.isWindows) {
// Device Information
await _peripheralManager.addService(
GATTService(
uuid: UUID.fromString('180A'),
isPrimary: true,
characteristics: [
GATTCharacteristic.immutable(
uuid: UUID.fromString('2A29'),
value: Uint8List.fromList('BikeControl'.codeUnits),
descriptors: [],
),
GATTCharacteristic.immutable(
uuid: UUID.fromString('2A25'),
value: Uint8List.fromList('09-B48123283828F1337'.codeUnits),
descriptors: [],
),
GATTCharacteristic.immutable(
uuid: UUID.fromString('2A27'),
value: Uint8List.fromList('A.0'.codeUnits),
descriptors: [],
),
GATTCharacteristic.immutable(
uuid: UUID.fromString('2A26'),
value: Uint8List.fromList((packageInfoValue?.version ?? '1.0.0').codeUnits),
descriptors: [],
),
],
includedServices: [],
),
);
}
// Battery Service
await peripheralManager.addService(
await _peripheralManager.addService(
GATTService(
uuid: UUID.fromString('180F'),
isPrimary: true,
@@ -239,9 +235,9 @@ class ZwiftEmulator {
);
// Unknown Service
await peripheralManager.addService(
await _peripheralManager.addService(
GATTService(
uuid: UUID.fromString(ZwiftConstants.ZWIFT_RIDE_CUSTOM_SERVICE_UUID_SHORT),
uuid: UUID.fromString(ZwiftConstants.ZWIFT_CUSTOM_SERVICE_UUID),
isPrimary: true,
characteristics: [
_asyncCharacteristic!,
@@ -253,7 +249,7 @@ class ZwiftEmulator {
],
permissions: [],
),
syncTxCharacteristic,
_syncTxCharacteristic!,
GATTCharacteristic.mutable(
uuid: UUID.fromString('00000005-19CA-4651-86E5-FA29DCDD09D1'),
descriptors: [],
@@ -284,9 +280,9 @@ class ZwiftEmulator {
}
final advertisement = Advertisement(
name: 'SwiftControl',
name: 'KICKR BIKE PRO 1337',
serviceUUIDs: [UUID.fromString(ZwiftConstants.ZWIFT_RIDE_CUSTOM_SERVICE_UUID_SHORT)],
serviceData: {
/*serviceData: {
UUID.fromString(ZwiftConstants.ZWIFT_RIDE_CUSTOM_SERVICE_UUID_SHORT): Uint8List.fromList([0x02]),
},
manufacturerSpecificData: [
@@ -294,24 +290,34 @@ class ZwiftEmulator {
id: 0x094A,
data: Uint8List.fromList([ZwiftConstants.CLICK_V2_LEFT_SIDE, 0x13, 0x37]),
),
],
],*/
);
print('Starting advertising with Zwift service...');
await peripheralManager.startAdvertising(advertisement);
_isAdvertising = true;
await _peripheralManager.startAdvertising(advertisement);
_isLoading = false;
onUpdate();
}
Future<void> stopAdvertising() async {
await peripheralManager.stopAdvertising();
_isAdvertising = false;
await _peripheralManager.stopAdvertising();
isStarted.value = false;
isConnected.value = false;
_isLoading = false;
}
Future<String> sendAction(InGameAction inGameAction, int? inGameActionValue) async {
final button = switch (inGameAction) {
Future<void> _sendKeepAlive() async {
await Future.delayed(const Duration(seconds: 5));
if (isConnected.value && _central != null) {
final zero = Uint8List.fromList([Opcode.CONTROLLER_NOTIFICATION.value, 0x08, 0xFF, 0xFF, 0xFF, 0xFF, 0x0F]);
_peripheralManager.notifyCharacteristic(_central!, _syncTxCharacteristic!, value: zero);
_sendKeepAlive();
}
}
@override
Future<ActionResult> sendAction(KeyPair keyPair, {required bool isKeyDown, required bool isKeyUp}) async {
final button = switch (keyPair.inGameAction) {
InGameAction.shiftUp => RideButtonMask.SHFT_UP_R_BTN,
InGameAction.shiftDown => RideButtonMask.SHFT_UP_L_BTN,
InGameAction.uturn => RideButtonMask.DOWN_BTN,
@@ -326,7 +332,7 @@ class ZwiftEmulator {
};
if (button == null) {
return 'Action ${inGameAction.name} not supported by Zwift Emulator';
return NotHandled('Action ${keyPair.inGameAction!.name} not supported by Zwift Emulator');
}
final status = RideKeyPadStatus()
@@ -335,33 +341,135 @@ class ZwiftEmulator {
final bytes = status.writeToBuffer();
final commandProto = Uint8List.fromList([
Opcode.CONTROLLER_NOTIFICATION.value,
...bytes,
]);
if (isKeyDown) {
final commandProto = Uint8List.fromList([
Opcode.CONTROLLER_NOTIFICATION.value,
...bytes,
]);
peripheralManager.notifyCharacteristic(_central!, _asyncCharacteristic!, value: commandProto);
_peripheralManager.notifyCharacteristic(
_central!,
_asyncCharacteristic!,
value: commandProto,
);
}
final zero = Uint8List.fromList([0x23, 0x08, 0xFF, 0xFF, 0xFF, 0xFF, 0x0F]);
peripheralManager.notifyCharacteristic(_central!, _asyncCharacteristic!, value: zero);
return 'Sent action: ${inGameAction.name}';
if (isKeyUp) {
final zero = Uint8List.fromList([Opcode.CONTROLLER_NOTIFICATION.value, 0x08, 0xFF, 0xFF, 0xFF, 0xFF, 0x0F]);
_peripheralManager.notifyCharacteristic(_central!, _asyncCharacteristic!, value: zero);
}
return Success('Sent action: ${keyPair.inGameAction!.name}');
}
}
class ZwiftEmulatorInformation extends StatelessWidget {
const ZwiftEmulatorInformation({super.key});
@override
Widget build(BuildContext context) {
return ValueListenableBuilder(
valueListenable: zwiftEmulator.isConnected,
builder: (context, isConnected, _) {
return StatefulBuilder(
builder: (context, setState) {
return Text('Zwift is ${isConnected ? 'connected' : 'not connected'}');
},
);
},
Uint8List? handleWriteRequest(String characteristic, Uint8List value) {
print(
'Write request for characteristic: $characteristic',
);
switch (characteristic.toUpperCase()) {
case ZwiftConstants.ZWIFT_SYNC_RX_CHARACTERISTIC_UUID:
print(
'Handling write request for SYNC RX characteristic, value: ${value.map((e) => e.toRadixString(16).padLeft(2, '0')).join(' ')}\n${String.fromCharCodes(value)}',
);
Opcode? opcode = Opcode.valueOf(value[0]);
Uint8List message = value.sublist(1);
switch (opcode) {
case Opcode.RIDE_ON:
print('Sending handshake');
return ZwiftConstants.RIDE_ON;
case Opcode.GET:
final response = Get.fromBuffer(message);
final dataObjectType = DO.valueOf(response.dataObjectId);
print('Received GET for data object: $dataObjectType');
switch (dataObjectType) {
case DO.PAGE_DEV_INFO:
/*final devInfo = DevInfoPage(
deviceName: 'Zwift Click'.codeUnits,
deviceUid: '0B-58D15ABB4363'.codeUnits,
manufacturerId: 0x01,
serialNumber: '58D15ABB4363'.codeUnits,
protocolVersion: 515,
systemFwVersion: [0, 0, 1, 1],
productId: 11,
systemHwRevision: 'B.0'.codeUnits,
deviceCapabilities: [DevInfoPage_DeviceCapabilities(deviceType: 2, capabilities: 1)],
);
final serverInfoResponse = Uint8List.fromList([
Opcode.GET_RESPONSE.value,
...GetResponse(
dataObjectId: DO.PAGE_DEV_INFO.value,
dataObjectData: devInfo.writeToBuffer(),
).writeToBuffer(),
]);*/
// 3C080012460A440883041204000001011A0B5A7769667420436C69636B320F30422D3538443135414242343336333A03422E304204080210014801500B5A0C353844313541424234333633
final expected = Uint8List.fromList(
hexToBytes(
'3C080012460A440883041204000001011A0B5A7769667420436C69636B320F30422D3538443135414242343336333A03422E304204080210014801500B5A0C353844313541424234333633',
),
);
return expected;
case DO.PAGE_CLIENT_SERVER_CONFIGURATION:
final response = Uint8List.fromList([
Opcode.GET_RESPONSE.value,
...GetResponse(
dataObjectId: DO.PAGE_CLIENT_SERVER_CONFIGURATION.value,
dataObjectData: ClientServerCfgPage(
notifications: 0,
).writeToBuffer(),
).writeToBuffer(),
]);
return response;
case DO.PAGE_CONTROLLER_INPUT_CONFIG:
final response = Uint8List.fromList([
Opcode.GET_RESPONSE.value,
...GetResponse(
dataObjectId: DO.PAGE_CONTROLLER_INPUT_CONFIG.value,
dataObjectData: ControllerInputConfigPage(
supportedDigitalInputs: 4607,
supportedAnalogInputs: 0,
analogDeadZone: [],
analogInputRange: [],
).writeToBuffer(),
).writeToBuffer(),
]);
return response;
case DO.BATTERY_STATE:
final response = Uint8List.fromList([
Opcode.GET_RESPONSE.value,
...GetResponse(
dataObjectId: DO.BATTERY_STATE.value,
dataObjectData: BatteryStatus(
chgState: ChargingState.CHARGING_IDLE,
percLevel: 100,
timeToEmpty: 0,
timeToFull: 0,
).writeToBuffer(),
).writeToBuffer(),
]);
return response;
default:
print('Unhandled data object type for GET: $dataObjectType');
}
break;
}
break;
default:
print('Unhandled write request for characteristic: $characteristic $value');
}
return null;
}
void cleanup() {
_peripheralManager.stopAdvertising();
_peripheralManager.removeAllServices();
_isServiceAdded = false;
_isSubscribedToEvents = false;
_central = null;
isConnected.value = false;
isStarted.value = false;
_isLoading = false;
}
}

View File

@@ -1,8 +1,11 @@
import 'package:flutter/foundation.dart';
import 'package:swift_control/bluetooth/devices/zwift/constants.dart';
import 'package:swift_control/bluetooth/devices/zwift/protocol/zwift.pb.dart';
import 'package:swift_control/bluetooth/devices/zwift/zwift_device.dart';
import 'package:swift_control/utils/keymap/buttons.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
import 'package:bike_control/bluetooth/devices/zwift/constants.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zwift.pb.dart';
import 'package:bike_control/bluetooth/devices/zwift/zwift_device.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/i18n_extension.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
class ZwiftPlay extends ZwiftDevice {
ZwiftPlay(super.scanResult)
@@ -28,6 +31,9 @@ class ZwiftPlay extends ZwiftDevice {
@override
List<int> get startCommand => ZwiftConstants.RIDE_ON + ZwiftConstants.RESPONSE_START_PLAY;
@override
bool get canVibrate => true;
@override
List<ControllerButton> processClickNotification(Uint8List message) {
final status = PlayKeyPadStatus.fromBuffer(message);
@@ -56,4 +62,23 @@ class ZwiftPlay extends ZwiftDevice {
@override
String get latestFirmwareVersion => '1.3.1';
@override
Widget showInformation(BuildContext context) {
return Column(
spacing: 16,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
super.showInformation(context),
Checkbox(
trailing: Expanded(child: Text(context.i18n.enableVibrationFeedback)),
state: core.settings.getVibrationEnabled() ? CheckboxState.checked : CheckboxState.unchecked,
onChanged: (value) async {
await core.settings.setVibrationEnabled(value == CheckboxState.checked);
},
),
],
);
}
}

View File

@@ -1,14 +1,14 @@
import 'package:dartx/dartx.dart';
import 'package:flutter/foundation.dart';
import 'package:protobuf/protobuf.dart' as $pb;
import 'package:swift_control/bluetooth/devices/zwift/constants.dart';
import 'package:swift_control/bluetooth/devices/zwift/protocol/zp.pb.dart';
import 'package:swift_control/bluetooth/devices/zwift/protocol/zp_vendor.pb.dart';
import 'package:swift_control/bluetooth/devices/zwift/protocol/zwift.pb.dart';
import 'package:swift_control/bluetooth/devices/zwift/zwift_device.dart';
import 'package:swift_control/bluetooth/messages/notification.dart';
import 'package:swift_control/main.dart';
import 'package:swift_control/utils/keymap/buttons.dart';
import 'package:bike_control/bluetooth/devices/zwift/constants.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zp.pb.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zp_vendor.pb.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zwift.pb.dart';
import 'package:bike_control/bluetooth/devices/zwift/zwift_device.dart';
import 'package:bike_control/bluetooth/messages/notification.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:universal_ble/universal_ble.dart';
class ZwiftRide extends ZwiftDevice {
@@ -44,10 +44,10 @@ class ZwiftRide extends ZwiftDevice {
);
@override
String get customServiceId => ZwiftConstants.ZWIFT_RIDE_CUSTOM_SERVICE_UUID;
String get latestFirmwareVersion => '1.2.0';
@override
String get latestFirmwareVersion => '1.2.0';
bool get canVibrate => true;
@override
Future<void> processData(Uint8List bytes) async {
@@ -146,7 +146,7 @@ class ZwiftRide extends ZwiftDevice {
final notification = BatteryNotification.fromBuffer(message);
if (batteryLevel != notification.newPercLevel) {
batteryLevel = notification.newPercLevel;
connection.signalChange(this);
core.connection.signalChange(this);
}
break;
case Opcode.CONTROLLER_NOTIFICATION:
@@ -230,7 +230,7 @@ class ZwiftRide extends ZwiftDevice {
}
await UniversalBle.write(
device.deviceId,
customServiceId,
customService!.uuid,
syncRxCharacteristic!.uuid,
buffer,
withoutResponse: true,
@@ -244,7 +244,7 @@ class ZwiftRide extends ZwiftDevice {
}
await UniversalBle.write(
device.deviceId,
customServiceId,
customService!.uuid,
syncRxCharacteristic!.uuid,
buffer,
withoutResponse: true,

View File

@@ -1,13 +1,20 @@
import 'package:bike_control/bluetooth/devices/zwift/protocol/zp.pb.dart';
import 'package:bike_control/utils/actions/base_actions.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:bike_control/widgets/keymap_explanation.dart';
import 'package:dartx/dartx.dart';
import 'package:swift_control/utils/keymap/buttons.dart';
import 'package:swift_control/widgets/keymap_explanation.dart';
import 'package:flutter/foundation.dart';
class BaseNotification {}
class LogNotification extends BaseNotification {
final String message;
LogNotification(this.message);
LogNotification(this.message) {
if (kDebugMode) {
print('LogNotification: $message');
}
}
@override
String toString() {
@@ -46,3 +53,28 @@ class ButtonNotification extends BaseNotification {
@override
int get hashCode => buttonsClicked.hashCode;
}
class ActionNotification extends BaseNotification {
final ActionResult result;
ActionNotification(this.result);
@override
String toString() {
return result.message;
}
}
class AlertNotification extends LogNotification {
final LogLevel level;
final String alertMessage;
final VoidCallback? onTap;
final String? buttonTitle;
AlertNotification(this.level, this.alertMessage, {this.onTap, this.buttonTitle}) : super(alertMessage);
@override
String toString() {
return 'Warning: $alertMessage';
}
}

View File

@@ -0,0 +1,310 @@
import 'dart:io';
import 'package:bike_control/bluetooth/devices/trainer_connection.dart';
import 'package:bike_control/bluetooth/devices/zwift/protocol/zp.pb.dart';
import 'package:bike_control/bluetooth/messages/notification.dart';
import 'package:bike_control/gen/l10n.dart';
import 'package:bike_control/utils/actions/base_actions.dart';
import 'package:bike_control/utils/actions/remote.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:bike_control/utils/requirements/multi.dart';
import 'package:bluetooth_low_energy/bluetooth_low_energy.dart';
import 'package:flutter/foundation.dart';
import 'package:permission_handler/permission_handler.dart';
import '../utils/keymap/keymap.dart';
class RemotePairing extends TrainerConnection {
bool get isLoading => _isLoading;
late final _peripheralManager = PeripheralManager();
bool _isLoading = false;
bool _isServiceAdded = false;
bool _isSubscribedToEvents = false;
Central? _central;
GATTCharacteristic? _inputReport;
static const String connectionTitle = 'Remote Control';
RemotePairing()
: super(
title: connectionTitle,
supportedActions: InGameAction.values,
);
Future<void> reconnect() async {
await _peripheralManager.stopAdvertising();
await _peripheralManager.removeAllServices();
_isServiceAdded = false;
startAdvertising().catchError((e) {
core.settings.setRemoteControlEnabled(false);
core.connection.signalNotification(
AlertNotification(LogLevel.LOGLEVEL_WARNING, 'Failed to start Remote Control pairing: $e'),
);
});
}
Future<void> startAdvertising() async {
_isLoading = true;
isStarted.value = true;
_peripheralManager.stateChanged.forEach((state) {
print('Peripheral manager state: ${state.state}');
});
if (!kIsWeb && Platform.isAndroid) {
_peripheralManager.connectionStateChanged.forEach((state) {
print('Peripheral connection state: ${state.state} of ${state.central.uuid}');
if (state.state == ConnectionState.connected) {
} else if (state.state == ConnectionState.disconnected) {
_central = null;
isConnected.value = false;
core.connection.signalNotification(
AlertNotification(LogLevel.LOGLEVEL_INFO, AppLocalizations.current.disconnected),
);
}
});
final status = await Permission.bluetoothAdvertise.request();
if (!status.isGranted) {
print('Bluetooth advertise permission not granted');
isStarted.value = false;
return;
}
}
while (_peripheralManager.state != BluetoothLowEnergyState.poweredOn && core.settings.getRemoteControlEnabled()) {
print('Waiting for peripheral manager to be powered on...');
if (core.settings.getLastTarget() == Target.thisDevice) {
return;
}
await Future.delayed(Duration(seconds: 1));
}
final inputReport = GATTCharacteristic.mutable(
uuid: UUID.fromString('2A4D'),
permissions: [GATTCharacteristicPermission.read],
properties: [GATTCharacteristicProperty.notify, GATTCharacteristicProperty.read],
descriptors: [
GATTDescriptor.immutable(
// Report Reference: ID=1, Type=Input(1)
uuid: UUID.fromString('2908'),
value: Uint8List.fromList([0x01, 0x01]),
),
],
);
if (!_isServiceAdded) {
await Future.delayed(Duration(seconds: 1));
final reportMapDataAbsolute = Uint8List.fromList([
0x05, 0x01, // Usage Page (Generic Desktop)
0x09, 0x02, // Usage (Mouse)
0xA1, 0x01, // Collection (Application)
0x85, 0x01, // Report ID (1)
0x09, 0x01, // Usage (Pointer)
0xA1, 0x00, // Collection (Physical)
0x05, 0x09, // Usage Page (Button)
0x19, 0x01, // Usage Min (1)
0x29, 0x03, // Usage Max (3)
0x15, 0x00, // Logical Min (0)
0x25, 0x01, // Logical Max (1)
0x95, 0x03, // Report Count (3)
0x75, 0x01, // Report Size (1)
0x81, 0x02, // Input (Data,Var,Abs) // buttons
0x95, 0x01, // Report Count (1)
0x75, 0x05, // Report Size (5)
0x81, 0x03, // Input (Const,Var,Abs) // padding
0x05, 0x01, // Usage Page (Generic Desktop)
0x09, 0x30, // Usage (X)
0x09, 0x31, // Usage (Y)
0x15, 0x00, // Logical Min (0)
0x25, 0x64, // Logical Max (100)
0x75, 0x08, // Report Size (8)
0x95, 0x02, // Report Count (2)
0x81, 0x02, // Input (Data,Var,Abs)
0xC0,
0xC0,
]);
// 1) Build characteristics
final hidInfo = GATTCharacteristic.immutable(
uuid: UUID.fromString('2A4A'),
value: Uint8List.fromList([0x11, 0x01, 0x00, 0x02]),
descriptors: [], // HID v1.11, country=0, flags=2
);
final reportMap = GATTCharacteristic.immutable(
uuid: UUID.fromString('2A4B'),
//properties: [GATTCharacteristicProperty.read],
//permissions: [GATTCharacteristicPermission.read],
value: reportMapDataAbsolute,
descriptors: [
GATTDescriptor.immutable(uuid: UUID.fromString('2908'), value: Uint8List.fromList([0x0, 0x0])),
],
);
final protocolMode = GATTCharacteristic.mutable(
uuid: UUID.fromString('2A4E'),
properties: [GATTCharacteristicProperty.read, GATTCharacteristicProperty.writeWithoutResponse],
permissions: [GATTCharacteristicPermission.read, GATTCharacteristicPermission.write],
descriptors: [],
);
final hidControlPoint = GATTCharacteristic.mutable(
uuid: UUID.fromString('2A4C'),
properties: [GATTCharacteristicProperty.writeWithoutResponse],
permissions: [GATTCharacteristicPermission.write],
descriptors: [],
);
// Input report characteristic (notify)
final keyboardInputReport = GATTCharacteristic.mutable(
uuid: UUID.fromString('2A4D'),
permissions: [GATTCharacteristicPermission.read],
properties: [GATTCharacteristicProperty.notify, GATTCharacteristicProperty.read],
descriptors: [
GATTDescriptor.immutable(
// Report Reference: ID=1, Type=Input(1)
uuid: UUID.fromString('2908'),
value: Uint8List.fromList([0x02, 0x01]),
),
],
);
final outputReport = GATTCharacteristic.mutable(
uuid: UUID.fromString('2A4D'),
permissions: [GATTCharacteristicPermission.read, GATTCharacteristicPermission.write],
properties: [
GATTCharacteristicProperty.read,
GATTCharacteristicProperty.write,
GATTCharacteristicProperty.writeWithoutResponse,
],
descriptors: [
GATTDescriptor.immutable(
// Report Reference: ID=1, Type=Input(1)
uuid: UUID.fromString('2908'),
value: Uint8List.fromList([0x02, 0x02]),
),
],
);
// 2) HID service
final hidService = GATTService(
uuid: UUID.fromString(Platform.isIOS ? '1812' : '00001812-0000-1000-8000-00805F9B34FB'),
isPrimary: true,
characteristics: [
hidInfo,
reportMap,
protocolMode,
outputReport,
hidControlPoint,
keyboardInputReport,
inputReport,
],
includedServices: [],
);
if (!_isSubscribedToEvents) {
_isSubscribedToEvents = true;
_peripheralManager.characteristicReadRequested.forEach((char) {
print('Read request for characteristic: ${char}');
// You can respond to read requests here if needed
});
_peripheralManager.characteristicNotifyStateChanged.forEach((char) {
if (char.characteristic.uuid == inputReport.uuid) {
if (char.state) {
_inputReport = char.characteristic;
_central = char.central;
} else {
_inputReport = null;
_central = null;
}
}
print(
'Notify state changed for characteristic: ${char.characteristic.uuid} vs ${char.characteristic.uuid == inputReport.uuid}: ${char.state}',
);
});
}
await _peripheralManager.addService(hidService);
// 3) Optional Battery service
await _peripheralManager.addService(
GATTService(
uuid: UUID.fromString('180F'),
isPrimary: true,
characteristics: [
GATTCharacteristic.immutable(
uuid: UUID.fromString('2A19'),
value: Uint8List.fromList([100]),
descriptors: [],
),
],
includedServices: [],
),
);
_isServiceAdded = true;
}
final advertisement = Advertisement(
name:
'BikeControl ${Platform.isIOS
? 'iOS'
: Platform.isAndroid
? 'Android'
: ''}',
serviceUUIDs: [UUID.fromString(Platform.isIOS ? '1812' : '00001812-0000-1000-8000-00805F9B34FB')],
);
print('Starting advertising with Zwift service...');
await _peripheralManager.startAdvertising(advertisement);
_isLoading = false;
}
Future<void> stopAdvertising() async {
await _peripheralManager.stopAdvertising();
isStarted.value = false;
isConnected.value = false;
_isLoading = false;
}
Future<void> notifyCharacteristic(Uint8List value) async {
if (_inputReport != null && _central != null) {
await _peripheralManager.notifyCharacteristic(_central!, _inputReport!, value: value);
}
}
@override
Future<ActionResult> sendAction(KeyPair keyPair, {required bool isKeyDown, required bool isKeyUp}) async {
final point = await (core.actionHandler as RemoteActions).resolveTouchPosition(keyPair: keyPair, windowInfo: null);
final point2 = point; //Offset(100, 99.0);
await sendAbsMouseReport(0, point2.dx.toInt(), point2.dy.toInt());
await sendAbsMouseReport(1, point2.dx.toInt(), point2.dy.toInt());
await sendAbsMouseReport(0, point2.dx.toInt(), point2.dy.toInt());
return Success('Mouse clicked at: ${point2.dx.toInt()} ${point2.dy.toInt()}');
}
Uint8List absMouseReport(int buttons3bit, int x, int y) {
final b = buttons3bit & 0x07;
final xi = x.clamp(0, 100);
final yi = y.clamp(0, 100);
return Uint8List.fromList([b, xi, yi]);
}
// Send a relative mouse move + button state as 3-byte report: [buttons, dx, dy]
Future<void> sendAbsMouseReport(int buttons, int dx, int dy) async {
final bytes = absMouseReport(buttons, dx, dy);
if (kDebugMode) {
print('Preparing to send abs mouse report: buttons=$buttons, dx=$dx, dy=$dy');
print('Sending abs mouse report: ${bytes.map((e) => e.toRadixString(16).padLeft(2, '0'))}');
}
await notifyCharacteristic(bytes);
// we don't want to overwhelm the target device
await Future.delayed(Duration(milliseconds: 10));
}
}

427
lib/i10n/intl_de.arb Normal file
View File

@@ -0,0 +1,427 @@
{
"accessibilityDescription": "BikeControl benötigt Zugriffsberechtigungen, um Ihre Trainings-Apps zu steuern.",
"accessibilityDisclaimer": "BikeControl greift nur auf Ihren Bildschirm zu, um die von Ihnen konfigurierten Gesten auszuführen. Es werden keine weiteren Bedienungshilfen oder persönlichen Daten abgerufen.",
"accessibilityReasonControl": "• Um Ihnen die Steuerung von Apps wie MyWhoosh, IndieVelo und anderen über Ihre Zwift-Geräte zu ermöglichen",
"accessibilityReasonTouch": "• Um Berührungsgesten auf Ihrem Bildschirm zur Steuerung von Trainer-Apps zu simulieren.",
"accessibilityReasonWindow": "• Um zu erkennen, welches Trainings-App-Fenster aktuell aktiv ist",
"accessibilityServiceExplanation": "BikeControl benötigt die AccessibilityService API von Android, um ordnungsgemäß zu funktionieren.",
"accessibilityServiceNotRunning": "Der Bedienungshilfe-Dienst ist nicht verfügbar. \nFolge den Anweisungen unter",
"accessibilityServicePermissionRequired": "Berechtigung für barrierefreie Dienste erforderlich",
"accessibilityUsageGestures": "• Wenn Du die Tasten auf Deinem Zwift Click-, Zwift Ride- oder Zwift Play-Geräten drückst, simuliert BikeControl Berührungsgesten an bestimmten Bildschirmpositionen.",
"accessibilityUsageMonitor": "• Die App überwacht, welches Trainings-App-Fenster aktiv ist, um sicherzustellen, dass Gesten an die richtige App gesendet werden.",
"accessibilityUsageNoData": "• Über diesen Dienst werden keine personenbezogenen Daten abgerufen oder erfasst.",
"accessories": "Zubehör",
"action": "Aktion",
"adjustControllerButtons": "Controller-Tasten anpassen",
"allow": "Erlauben",
"allowAccessibilityService": "Barrierefreiheitsdienst zulassen",
"allowBluetoothConnections": "Bluetooth-Verbindungen zulassen",
"allowBluetoothScan": "Bluetooth-Scan zulassen",
"allowLocationForBluetooth": "Standortzugriff erlauben, damit Bluetooth-Scan funktioniert",
"allowPersistentNotification": "Benachrichtigungen zulassen",
"allowsRunningInBackground": "Ermöglicht es BikeControl, im Hintergrund weiterzulaufen.",
"appIdActions": "{appId} Aktionen",
"@appIdActions": {
"placeholders": {
"appId": {
"type": "String"
}
}
},
"battery": "Batterie",
"bluetoothAdvertiseAccess": "Bluetooth-Zugriff",
"bluetoothTurnedOn": "Bluetooth ist eingeschaltet",
"browserNotSupported": "Dieser Browser unterstützt kein Web-Bluetooth und die Plattform wird nicht unterstützt :(",
"button": "Taste.",
"cancel": "Abbrechen",
"changelog": "Änderungsprotokoll",
"checkMyWhooshConnectionScreen": "Überprüfe den Verbindungsbildschirm in MyWhoosh, um zu sehen, ob „Link“ verbunden ist.",
"chooseAnotherScreenshot": "Wähle einen anderen Screenshot aus",
"chooseBikeControlInConnectionScreen": "Wähle im Verbindungsbildschirm BikeControl aus.",
"clickAButtonOnYourController": "Klicke eine Controller-Taste, um deren Aktion zu bearbeiten, oder tippe auf das Bearbeitungssymbol.",
"clickV2EventInfo": "Dein Click V2 sendet möglicherweise keine Tastenereignisse mehr. Probier mal ein paar Tasten aus und schau, ob sie in BikeControl angezeigt werden.",
"clickV2Instructions": "Damit dein Zwift Click V2 optimal funktioniert, solltest du vor jeder Trainings-Session in der Zwift-App verbinden.\nWenn du das nicht machst, funktioniert der Click V2 nach einer Minute nicht mehr.\n\n1. Öffne die Zwift-App.\n2. Melde dich an (kein Abonnement nötig) und öffne den Bildschirm für die Geräteverbindung.\n3. Verbinde deinen Trainer und dann den Zwift Click V2.\n4. Schließe die Zwift-App wieder und verbinde dich erneut in BikeControl.",
"close": "Schließen",
"commandsRemainingToday": "{commandsRemainingToday}/{dailyCommandLimit} verbleibende Befehle heute",
"configuration": "Konfiguration",
"connectControllerToPreview": "Schließe ein Controller-Gerät an, um die Tastaturbelegung in der Vorschau anzuzeigen und anzupassen.",
"connectControllers": "Controller verbinden",
"connectDirectlyOverNetwork": "Direkte Verbindung über das Netzwerk",
"connectToTrainerApp": "Mit der Trainer-App verbinden",
"connectUsingBluetooth": "Verbindung über Bluetooth herstellen",
"connectUsingMyWhooshLink": "Verbinden über MyWhoosh „Link“",
"connected": "Verbunden",
"connectedControllers": "Verbundene Controller",
"connectedTo": "Verbunden mit {appId}",
"@connectedTo": {
"placeholders": {
"appId": {
"type": "String"
}
}
},
"connection": "Verbindung",
"continueAction": "Weitermachen",
"controlAppUsingModes": "Steuere {appName} über {modes}",
"@controlAppUsingModes": {
"placeholders": {
"appName": {
"type": "String"
},
"modes": {
"type": "String"
}
}
},
"controllers": "Controller",
"couldNotPerformButtonnamesplitbyuppercaseNoKeymapSet": "{button} konnte nicht ausgeführt werden: Keine Konfiguration festgelegt",
"create": "Erstellen",
"createNewKeymap": "Neue Tastaturbelegung erstellen",
"createNewProfileByDuplicating": "Erstelle ein neues benutzerdefiniertes Profil, indem „{profileName} “ dupliziert wird.",
"@createNewProfileByDuplicating": {
"placeholders": {
"profileName": {
"type": "String"
}
}
},
"createdNewCustomProfile": "Ein neues benutzerdefiniertes Profil wurde erstellt: {profileName}",
"@createdNewCustomProfile": {
"placeholders": {
"profileName": {
"type": "String"
}
}
},
"customizeControllerButtons": "Controller-Tasten anpassen für {appName}",
"@customizeControllerButtons": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"customizeKeymapHint": "Passe die Tastaturbelegung an, falls Probleme auftreten (z. B. falsche Tastatureingaben oder falsch ausgerichtete Touch-Positionen).",
"dailyCommandLimitReachedNotification": "Das tägliche Befehlslimit wurde für heute erreicht. Führe ein Upgrade durch, um die Vollversion mit unbegrenzten Befehlen freizuschalten.",
"dailyLimitReached": "Tageslimit erreicht ({dailyCommandCount} / {dailyCommandLimit} verbraucht)",
"delete": "Löschen",
"deleteProfile": "Profil löschen",
"deleteProfileConfirmation": "„{profileName} “ wirklich löschen? Diese Aktion kann nicht rückgängig gemacht werden.",
"@deleteProfileConfirmation": {
"placeholders": {
"profileName": {
"type": "String"
}
}
},
"deny": "Verweigern",
"deviceButton": "{deviceName} Taste",
"@deviceButton": {
"placeholders": {
"deviceName": {
"type": "String"
}
}
},
"disconnectDevices": "Geräte trennen",
"disconnected": "Getrennt",
"donateByBuyingFromPlayStore": "App im Play Store kaufen",
"donateViaCreditCard": "per Kreditkarte, Google Pay, Apple Pay und anderen Zahlungsarten",
"donateViaPaypal": "via PayPal",
"download": "Herunterladen",
"dragToReposition": "Zum Verschieben ziehen",
"duplicate": "Duplikat",
"enableAutoRotation": "Aktiviere die automatische Drehung auf Ihrem Gerät, um sicherzustellen, dass die App ordnungsgemäß funktioniert.",
"enableBluetooth": "Bluetooth aktivieren",
"enableKeyboardAccessMessage": "Aktiviere im folgenden Bildschirm die Tastatursteuerung für BikeControl. Falls BikeControl nicht angezeigt wird, füge es bitte manuell hinzu.",
"enableKeyboardMouseControl": "Aktiviere die Tastatur- und Maussteuerung für eine bessere Interaktion mit {appName}. Sobald die Funktion aktiviert ist, sind keine weiteren Aktionen oder Verbindungen erforderlich. BikeControl sendet die Maus- oder Tastatureingaben direkt an {appName} weiter.",
"@enableKeyboardMouseControl": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"enableMediaKeyDetection": "Medientastenerkennung aktivieren",
"enablePairingProcess": "Kopplungsprozess aktivieren",
"enablePermissions": "Berechtigungen aktivieren",
"enableSteeringWithPhone": "Sensoren Ihres Telefons aktivieren z.B. zum Lenken",
"enableVibrationFeedback": "Vibrationsfeedback beim Gangwechsel aktivieren",
"enableZwiftControllerBluetooth": "Zwift Controller aktivieren (Bluetooth)",
"enableZwiftControllerNetwork": "Zwift Controller aktivieren (Netzwerk)",
"errorStartingMyWhooshLink": "Fehler beim Starten des MyWhoosh Link-Servers. Bitte stelle sicher, dass die App „MyWhoosh Link“ nicht bereits auf diesem Gerät ausgeführt wird.",
"errorStartingOpenBikeControlBluetoothServer": "Fehler beim Starten des OpenBikeControl Bluetooth-Servers.",
"errorStartingOpenBikeControlServer": "Fehler beim Starten des OpenBikeControl-Servers.",
"exportAction": "Export",
"failedToImportProfile": "Profilimport fehlgeschlagen. Ungültiges Format.",
"failedToUpdate": "Aktualisierung fehlgeschlagen: {error}",
"@failedToUpdate": {
"placeholders": {
"error": {
"type": "String"
}
}
},
"firmware": "Firmware",
"forceCloseToUpdate": "Schließen die App, um die neue Version zu verwenden.",
"fullVersion": "Vollversion",
"fullVersionDescription": "Die Vollversion beinhaltet: \n- Unbegrenzte Befehle pro Tag \n- Zugriff auf alle zukünftigen Updates \n- Kein Abonnement! Nur eine einmalige Gebühr :)",
"getSupport": "Unterstützung erhalten",
"gotIt": "Verstanden!",
"grant": "Gewähren",
"granted": "Gewährt",
"helpRequested": "Hilfe für BikeControl v{version} angefordert",
"@helpRequested": {
"placeholders": {
"version": {
"type": "String"
}
}
},
"howBikeControlUsesPermission": "Wie nutzt BikeControl diese Berechtigung?",
"ignoredDevices": "Ignorierte Geräte",
"importAction": "Import",
"importProfile": "Profil importieren",
"instructions": "Anleitung",
"jsonData": "JSON-Daten",
"keyboardAccess": "Tastaturzugriff",
"latestVersion": "aktuell: {version}",
"@latestVersion": {
"placeholders": {
"version": {
"type": "String"
}
}
},
"leaveAReview": "Hinterlasse eine Bewertung",
"letsAppConnectOverBluetooth": "Lässt {appName} die Verbindung zu BikeControl über Bluetooth herstellen.",
"@letsAppConnectOverBluetooth": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"letsAppConnectOverNetwork": "{appName} direkt über das Netzwerk verbinden. Wähle BikeControl im Verbindungsbildschirm aus.",
"@letsAppConnectOverNetwork": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"license": "Lizenz",
"licenseStatus": "Lizenzstatus",
"loadScreenshotForPlacement": "Lade einen Screenshot aus dem Spiel zur Platzierung hoch.",
"logViewer": "Protokollanzeige",
"logs": "Protokolle",
"logsAreAlsoAt": "Die Protokolle befinden sich auch unter",
"logsHaveBeenCopiedToClipboard": "Die Protokolle wurden in die Zwischenablage kopiert.",
"longPress": "langes\nDrücken",
"longPressMode": "Modus „Langes Drücken“ (statt Wiederholen)",
"mailSupportExplanation": "Die individuelle Unterstützung per E-Mail ist für mich sehr aufwendig.\n\nBitte nutze daher Reddit, Facebook oder GitHub für Fragen und Probleme, damit die gesamte Community davon profitieren kann.",
"manageIgnoredDevices": "Ignorierte Geräte verwalten",
"manageProfile": "Profil verwalten",
"manualyControllingButton": "{trainerApp} manuell steuern!",
"mediaKeyDetectionTooltip": "Aktiviere diese Option, damit BikeControl Bluetooth-Fernbedienungen erkennt. \nDazu muss BikeControl als Mediaplayer fungieren.",
"miuiDeviceDetected": "MIUI-Gerät erkannt",
"miuiDisableBatteryOptimization": "• Batterieoptimierung für BikeControl deaktivieren",
"miuiEnableAutostart": "• Aktiviere den automatischen Start für BikeControl.",
"miuiEnsureProperWorking": "Um sicherzustellen, dass BikeControl ordnungsgemäß funktioniert:",
"miuiLockInRecentApps": "• Sperre die App in den zuletzt verwendeten Apps",
"miuiWarningDescription": "Auf Ihrem Gerät läuft MIUI, das dafür bekannt ist, Hintergrunddienste und Bedienungshilfen aggressiv zu beenden.",
"moreInformation": "Weitere Informationen",
"mustChooseAllowOrDeny": "Diese Berechtigung muss entweder erteilt oder verweigert werden, um fortfahren zu können.",
"myWhooshDirectConnectAction": "MyWhoosh „Link”-Aktion",
"myWhooshDirectConnection": " z. B. mit MyWhoosh „Link”.",
"myWhooshLinkConnected": "MyWhoosh „Link“ verbunden",
"myWhooshLinkDescriptionLocal": "Verbinde dich direkt mit MyWhoosh über die „Link”-Methode. Unterstützte Aktionen sind unter anderem Schalten, Emotes und Richtungswechsel. Die MyWhoosh Link-Begleit-App darf dabei NICHT gleichzeitig laufen.",
"myWhooshLinkInfo": "Schau mal im Abschnitt zur Fehlerbehebung nach, wenn du Probleme hast. Eine deutlich zuverlässigere Verbindungsmethode kommt bald!",
"needHelpClickHelp": "Hilfe benötigt? Klicke auf",
"needHelpDontHesitate": "den Button oben und zögere nicht, uns zu kontaktieren.",
"newConnectionMethodAnnouncement": "{trainerApp} wird in Kürze deutlich bessere und zuverlässigere Verbindungsmethoden unterstützen!",
"newCustomProfile": "Neues benutzerdefiniertes Profil",
"newProfileName": "Neuer Profilname",
"newVersionAvailable": "Neue Version verfügbar",
"newVersionAvailableWithVersion": "Neue Version verfügbar: {version}",
"@newVersionAvailableWithVersion": {
"placeholders": {
"version": {
"type": "String"
}
}
},
"next": "Nächste",
"noActionAssigned": "Keine Maßnahmen zugewiesen",
"noActionAssignedForButton": "{button} konnte nicht ausgeführt werden: Keine Aktion zugewiesen",
"noConnectionMethodIsConnectedOrActive": "Es ist keine Verbindungsmethode verbunden oder aktiv.",
"noConnectionMethodSelected": "Keine Verbindungsmethode ausgewählt",
"noControllerConnected": "Keine Verbindung",
"noControllerUseCompanionMode": "Kein Controller? Nutze den Companion Mode",
"noIgnoredDevices": "Keine ignorierten Geräte.",
"noTrainerSelected": "Kein Trainer ausgewählt",
"notConnected": "Nicht verbunden",
"notificationDescription": "Dadurch bleibt die App im Hintergrund aktiv und informiert, wenn sich die Verbindung zu Geräten ändert.",
"ok": "OK",
"openBikeControlActions": "OpenBikeControl-Aktionen",
"openBikeControlAnnouncement": "Tolle Neuigkeiten! {trainerApp} unterstützt das OpenBikeControl-Protokoll für eine bestmögliche Benutzererfahrung.",
"openBikeControlConnection": " z. B. durch Verwendung einer OpenBikeControl-Verbindung",
"otherConnectionMethods": "Andere Verbindungsmethoden",
"pairingDescription": "Die Kopplung ermöglicht volle Anpassungsmöglichkeiten, funktioniert aber möglicherweise nicht auf allen Geräten.",
"pairingInstructions": "Gehe auf Deinem {targetName} in die Bluetooth-Einstellungen und suche nach BikeControl oder dem Namen Ihres Geräts. Wenn Du die Fernbedienungsfunktion nutzen möchtest, ist eine Kopplung erforderlich.",
"@pairingInstructions": {
"placeholders": {
"targetName": {
"type": "String"
}
}
},
"pairingInstructionsIOS": "Gehe auf Deinem iPad zu „Einstellungen“ > „Bedienungshilfen“ > „Berühren“ > „AssistiveTouch“ > „Zeigergeräte“ > „Geräte“ und koppel Dein Gerät. Stelle sicher, dass AssistiveTouch aktiviert ist.",
"pasteExportedJsonData": "Füge die exportierten JSON-Daten unten ein:",
"pathCopiedToClipboard": "Der Pfad wurde in die Zwischenablage kopiert.",
"permissionsRequired": "Damit BikeControl nach Geräten in der Nähe suchen und bei Verbindungsänderungen informieren kann, aktiviere bitte die folgenden Berechtigungen:",
"platformNotSupported": "{platform} wird nicht unterstützt :(",
"@platformNotSupported": {
"placeholders": {
"platform": {
"type": "String"
}
}
},
"platformRestrictionNotSupported": "Aufgrund von Plattformbeschränkungen wird dieses Szenario nicht unterstützt.",
"platformRestrictionOtherDevicesOnly": "Aufgrund von Plattformbeschränkungen wird das Kontrollieren von {appName} nur auf anderen Geräten unterstützt.",
"@platformRestrictionOtherDevicesOnly": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"playPause": "Wiedergabe/Pause",
"pleaseSelectAConnectionMethodFirst": "Wähle bitte zuerst in den Trainereinstellungen eine Verbindungsmethode aus.",
"predefinedAction": "Vordefinierte {appName} Aktion",
"@predefinedAction": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"pressButtonOnClickDevice": "Drücke eine Taste auf Ihrem Click-Gerät.",
"pressKeyToAssign": "Drücke eine Taste auf Ihrer Tastatur, um sie {buttonName} zuzuweisen.",
"@pressKeyToAssign": {
"placeholders": {
"buttonName": {
"type": "String"
}
}
},
"previous": "Vorherige",
"profileExportedToClipboard": "Profil „{profileName} “ in die Zwischenablage kopiert",
"@profileExportedToClipboard": {
"placeholders": {
"profileName": {
"type": "String"
}
}
},
"profileImportedSuccessfully": "Profil erfolgreich importiert",
"profileName": "Profilname",
"purchase": "Kaufen",
"recommendedConnectionMethods": "Empfohlene Verbindungsmethoden",
"removeFromIgnoredList": "Aus der Ignorierliste entfernen",
"rename": "Umbenennen",
"renameProfile": "Profil umbenennen",
"requirement": "Anforderung",
"reset": "Zurücksetzen",
"restart": "Neustart",
"runAppOnPlatformRemotely": "{appName} auf {platform} laufen lassen und es von diesem Gerät aus fernsteuern via {preferredConnection}.",
"@runAppOnPlatformRemotely": {
"placeholders": {
"appName": {
"type": "String"
},
"platform": {
"type": "String"
},
"preferredConnection": {
"type": "String"
}
}
},
"runAppOnThisDevice": "{appName} läuft auf diesem Gerät.",
"@runAppOnThisDevice": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"save": "Speichern",
"scan": "SCAN",
"scanningForDevices": "Suche nach Geräten... Stelle sicher, dass diese eingeschaltet und in Reichweite sind und nicht mit einem anderen Gerät verbunden sind.",
"selectKeymap": "Tastaturbelegung auswählen",
"selectTargetWhereAppRuns": "Ziel auswählen, auf dem {appName} läuft",
"@selectTargetWhereAppRuns": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"selectTrainerApp": "Trainer-App auswählen",
"selectTrainerAppAndTarget": "Trainer-App und Zielgerät auswählen",
"selectTrainerAppPlaceholder": "Trainer-App auswählen",
"setting": "Einstellung",
"setupTrainer": "Trainer einrichten",
"share": "Teilen",
"showDonation": "Zeige Deine Wertschätzung durch eine Spende.",
"showSupportedControllers": "Unterstützte Controller anzeigen",
"showTroubleshootingGuide": "Anleitung zur Fehlerbehebung anzeigen",
"signal": "Signal",
"simulateButtons": "Trainersteuerung",
"simulateKeyboardShortcut": "Tastenkombination simulieren",
"simulateMediaKey": "Medientaste simulieren",
"simulateTouch": "Berührung simulieren",
"stop": "Stoppen",
"targetOtherDevice": "Anderes Gerät",
"targetThisDevice": "Dieses Gerät",
"theFollowingPermissionsRequired": "Folgende Berechtigungen sind erforderlich:",
"touchAreaInstructions": "1. Erstelle einen Screenshot Ihrer App (z. B. innerhalb von MyWhoosh) im Querformat.\n2. Lade den Screenshot über die Schaltfläche unten.\n3. Die App wird automatisch auf Querformat eingestellt, um eine genaue Zuordnung zu gewährleisten.\n4. Drücke eine Taste auf Ihrem Click-Gerät, um einen Touch-Bereich zu erstellen.\n5. Ziehe die Touch-Bereiche an die gewünschte Position auf dem Screenshot.\n6. Speicher und schließe diesen Bildschirm.",
"touchSimulationForegroundMessage": "Um Berührungen zu simulieren, muss die App im Vordergrund bleiben.",
"trainer": "Trainer",
"trialDaysRemaining": "{trialDaysRemaining} verbleibende Tage",
"trialExpired": "Testphase abgelaufen. Befehle beschränkt auf {dailyCommandLimit} pro Tag.",
"trialPeriodActive": "Testphase aktiv - {trialDaysRemaining} verbleibende Tage",
"trialPeriodDescription": "Während der Testphase stehen unbegrenzt viele Befehle zur Verfügung. Nach Ablauf der Testphase sind die Befehle auf {dailyCommandLimit} pro Tag eingeschränkt.",
"troubleshootingGuide": "Leitfaden zur Fehlerbehebung",
"tryingToConnectAgain": "Verbinde erneut...",
"unassignAction": "Zuweisung aufheben",
"unlockFullVersion": "Vollversion freischalten",
"update": "Aktualisieren",
"useCustomKeymapForButton": "Verwende eine benutzerdefinierte Tastaturbelegung, um die",
"version": "Version {version}",
"@version": {
"placeholders": {
"version": {
"type": "String"
}
}
},
"viewDetailedInstructions": "Detaillierte Anweisungen ansehen",
"volumeDown": "Lautstärke verringern",
"volumeUp": "Lautstärke erhöhen",
"waiting": "Warten...",
"waitingForConnectionKickrBike": "Verbindung wird hergestellt. Wähle KICKR BIKE PRO im {appName} Verbindungs-Menü.",
"@waitingForConnectionKickrBike": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"whatsNew": "Was ist neu",
"whyPermissionNeeded": "Wozu wird diese Berechtigung benötigt?",
"zwiftControllerAction": "Zwift Controller-Aktion",
"zwiftControllerDescription": "Ermöglicht es BikeControl, als Zwift-kompatibler Controller zu fungieren."
}

427
lib/i10n/intl_en.arb Normal file
View File

@@ -0,0 +1,427 @@
{
"accessibilityDescription": "BikeControl needs accessibility permission to control your training apps.",
"accessibilityDisclaimer": "BikeControl will only access your screen to perform the gestures you configure. No other accessibility features or personal information will be accessed.",
"accessibilityReasonControl": "• To enable you to control apps like MyWhoosh, IndieVelo, and others using your Zwift devices",
"accessibilityReasonTouch": "• To simulate touch gestures on your screen for controlling trainer apps",
"accessibilityReasonWindow": "• To detect which training app window is currently active",
"accessibilityServiceExplanation": "BikeControl needs to use Android's AccessibilityService API to function properly.",
"accessibilityServiceNotRunning": "Accessibility Service is not running.\nFollow instructions at",
"accessibilityServicePermissionRequired": "Accessibility Service Permission Required",
"accessibilityUsageGestures": "• When you press buttons on your Zwift Click, Zwift Ride, or Zwift Play devices, BikeControl simulates touch gestures at specific screen locations",
"accessibilityUsageMonitor": "• The app monitors which training app window is active to ensure gestures are sent to the correct app",
"accessibilityUsageNoData": "• No personal data is accessed or collected through this service",
"accessories": "Accessories",
"action": "Action",
"adjustControllerButtons": "Adjust Controller Buttons",
"allow": "Allow",
"allowAccessibilityService": "Allow Accessibility Service",
"allowBluetoothConnections": "Allow Bluetooth Connections",
"allowBluetoothScan": "Allow Bluetooth Scan",
"allowLocationForBluetooth": "Allow Location so Bluetooth scan works",
"allowPersistentNotification": "Allow Notifications",
"allowsRunningInBackground": "Allows BikeControl to keep running in background",
"appIdActions": "{appId} actions",
"@appIdActions": {
"placeholders": {
"appId": {
"type": "String"
}
}
},
"battery": "Battery",
"bluetoothAdvertiseAccess": "Bluetooth Advertise access",
"bluetoothTurnedOn": "Bluetooth turned on",
"browserNotSupported": "This Browser does not support Web Bluetooth and platform is not supported :(",
"button": "button.",
"cancel": "Cancel",
"changelog": "Changelog",
"checkMyWhooshConnectionScreen": "Check the connection screen in MyWhoosh to see if \"Link\" is connected.",
"chooseAnotherScreenshot": "Choose another screenshot",
"chooseBikeControlInConnectionScreen": "Choose BikeControl in the connection screen.",
"clickAButtonOnYourController": "Click a button on your controller to edit its action or tap the edit icon.",
"clickV2EventInfo": "Your Click V2 may no longer send button events. Please check by tapping a few buttons and see if they are visible in BikeControl.",
"clickV2Instructions": "To make your Zwift Click V2 work best, you should connect it in the Zwift app before each training session.\nIf you don't do that, the Click V2 will stop working after a minute.\n\n1. Open Zwift app\n2. Log in (subscription not required) and open the device connection screen\n3. Connect your Trainer, then connect the Zwift Click V2\n4. Close the Zwift app again and connect again in BikeControl",
"close": "Close",
"commandsRemainingToday": "{commandsRemainingToday}/{dailyCommandLimit} commands remaining today",
"configuration": "Configuration",
"connectControllerToPreview": "Connect a controller device to preview and customize the keymap.",
"connectControllers": "Connect Controllers",
"connectDirectlyOverNetwork": "Connect directly over Network",
"connectToTrainerApp": "Connect to Trainer app",
"connectUsingBluetooth": "Connect using Bluetooth",
"connectUsingMyWhooshLink": "Connect using MyWhoosh \"Link\"",
"connected": "Connected",
"connectedControllers": "Connected Controllers",
"connectedTo": "Connected to {appId}",
"@connectedTo": {
"placeholders": {
"appId": {
"type": "String"
}
}
},
"connection": "Connection",
"continueAction": "Continue",
"controlAppUsingModes": "Control {appName} using {modes}",
"@controlAppUsingModes": {
"placeholders": {
"appName": {
"type": "String"
},
"modes": {
"type": "String"
}
}
},
"controllers": "Controllers",
"couldNotPerformButtonnamesplitbyuppercaseNoKeymapSet": "Could not perform {button}: No keymap set",
"create": "Create",
"createNewKeymap": "Create new keymap",
"createNewProfileByDuplicating": "Create new custom profile by duplicating \"{profileName}\"",
"@createNewProfileByDuplicating": {
"placeholders": {
"profileName": {
"type": "String"
}
}
},
"createdNewCustomProfile": "Created a new custom profile: {profileName}",
"@createdNewCustomProfile": {
"placeholders": {
"profileName": {
"type": "String"
}
}
},
"customizeControllerButtons": "Customize Controller buttons for {appName}",
"@customizeControllerButtons": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"customizeKeymapHint": "Customize the keymap if you experience any issues (e.g. wrong keyboard output, or misaligned touch placements)",
"dailyCommandLimitReachedNotification": "Daily command limit reached for today. Upgrade to unlock the full version with unlimited commands.",
"dailyLimitReached": "Daily limit reached ({dailyCommandCount}/{dailyCommandLimit} used)",
"delete": "Delete",
"deleteProfile": "Delete Profile",
"deleteProfileConfirmation": "Are you sure you want to delete \"{profileName}\"? This action cannot be undone.",
"@deleteProfileConfirmation": {
"placeholders": {
"profileName": {
"type": "String"
}
}
},
"deny": "Deny",
"deviceButton": "{deviceName} button",
"@deviceButton": {
"placeholders": {
"deviceName": {
"type": "String"
}
}
},
"disconnectDevices": "Disconnect Devices",
"disconnected": "Disconnected",
"donateByBuyingFromPlayStore": "by buying the app from Play Store",
"donateViaCreditCard": "via Credit Card, Google Pay, Apple Pay and others",
"donateViaPaypal": "via PayPal",
"download": "Download",
"dragToReposition": "Drag to reposition",
"duplicate": "Duplicate",
"enableAutoRotation": "Enable auto-rotation on your device to make sure the app works correctly.",
"enableBluetooth": "Enable Bluetooth",
"enableKeyboardAccessMessage": "Enable keyboard access in the following screen for BikeControl. If you don't see BikeControl, please add it manually.",
"enableKeyboardMouseControl": "Enable keyboard and mouse control for better interaction with {appName}. Once active, no further action or connection is needed. BikeControl will directly send the mouse or keyboard input to {appName}.",
"@enableKeyboardMouseControl": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"enableMediaKeyDetection": "Enable Media Key Detection",
"enablePairingProcess": "Enable Pairing Process",
"enablePermissions": "Enable Permissions",
"enableSteeringWithPhone": "Enable Phones' sensors to enable e.g. steering",
"enableVibrationFeedback": "Enable vibration feedback when shifting gears",
"enableZwiftControllerBluetooth": "Enable Zwift Controller (Bluetooth)",
"enableZwiftControllerNetwork": "Enable Zwift Controller (Network)",
"errorStartingMyWhooshLink": "Error starting MyWhoosh Link server. Please make sure the \"MyWhoosh Link\" app is not already running on this device.",
"errorStartingOpenBikeControlBluetoothServer": "Error starting OpenBikeControl Bluetooth server.",
"errorStartingOpenBikeControlServer": "Error starting OpenBikeControl server.",
"exportAction": "Export",
"failedToImportProfile": "Failed to import profile. Invalid format.",
"failedToUpdate": "Failed to update: {error}",
"@failedToUpdate": {
"placeholders": {
"error": {
"type": "String"
}
}
},
"firmware": "Firmware",
"forceCloseToUpdate": "Force-close the app to use the new version",
"fullVersion": "Full Version",
"fullVersionDescription": "The full version includes:\n- Unlimited commands per day\n- Access to all future updates\n- No subscription! A one-time fee only :)",
"getSupport": "Get Support",
"gotIt": "Got it!",
"grant": "Grant",
"granted": "Granted",
"helpRequested": "Help requested for BikeControl v{version}",
"@helpRequested": {
"placeholders": {
"version": {
"type": "String"
}
}
},
"howBikeControlUsesPermission": "How does BikeControl use this permission?",
"ignoredDevices": "Ignored Devices",
"importAction": "Import",
"importProfile": "Import Profile",
"instructions": "Instructions",
"jsonData": "JSON Data",
"keyboardAccess": "Keyboard access",
"latestVersion": "latest: {version}",
"@latestVersion": {
"placeholders": {
"version": {
"type": "String"
}
}
},
"leaveAReview": "Leave a Review",
"letsAppConnectOverBluetooth": "Lets {appName} connect to BikeControl over Bluetooth.",
"@letsAppConnectOverBluetooth": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"letsAppConnectOverNetwork": "Lets {appName} connect directly over the Network. Choose BikeControl in the connection screen.",
"@letsAppConnectOverNetwork": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"license": "License",
"licenseStatus": "License Status",
"loadScreenshotForPlacement": "Load in-game screenshot for placement",
"logViewer": "Log Viewer",
"logs": "Logs",
"logsAreAlsoAt": "Logs are also at",
"logsHaveBeenCopiedToClipboard": "Logs have been copied to clipboard",
"longPress": "long\npress",
"longPressMode": "Long Press Mode (vs. repeating)",
"mailSupportExplanation": "Providing individual support via email is a lot of work for me.\n\nPlease consider using Reddit, Facebook or GitHub for questions and issues so that the whole community can benefit from it.",
"manageIgnoredDevices": "Manage Ignored Devices",
"manageProfile": "Manage Profile",
"manualyControllingButton": "Control {trainerApp} manually!",
"mediaKeyDetectionTooltip": "Enable this option to allow BikeControl to detect bluetooth remotes.\nIn order to do so BikeControl needs to act as a media player.",
"miuiDeviceDetected": "MIUI Device Detected",
"miuiDisableBatteryOptimization": "• Disable battery optimization for BikeControl",
"miuiEnableAutostart": "• Enable autostart for BikeControl",
"miuiEnsureProperWorking": "To ensure BikeControl works properly:",
"miuiLockInRecentApps": "• Lock the app in recent apps",
"miuiWarningDescription": "Your device is running MIUI, which is known to aggressively kill background services and accessibility services.",
"moreInformation": "More Information",
"mustChooseAllowOrDeny": "You must choose to either Allow or Deny this permission to continue.",
"myWhooshDirectConnectAction": "MyWhoosh \"Link\" Action",
"myWhooshDirectConnection": " e.g. by using MyWhoosh \"Link\"",
"myWhooshLinkConnected": "MyWhoosh \"Link\" connected",
"myWhooshLinkDescriptionLocal": "Connect directly to MyWhoosh via the \"Link\" method. Supported actions are shifting, Emotes, turn directions, among others. The MyWhoosh Link companion app must NOT be running at the same time.",
"myWhooshLinkInfo": "Please check the troubleshooting section if you encounter any issues. A much more reliable connection method is coming soon!",
"needHelpClickHelp": "Need help? Click on the",
"needHelpDontHesitate": "button on top and don't hesitate to contact us.",
"newConnectionMethodAnnouncement": "{trainerApp} will soon support much better, reliable connection methods - stay tuned for updates!",
"newCustomProfile": "New Custom Profile",
"newProfileName": "New Profile Name",
"newVersionAvailable": "New version available",
"newVersionAvailableWithVersion": "New version available: {version}",
"@newVersionAvailableWithVersion": {
"placeholders": {
"version": {
"type": "String"
}
}
},
"next": "Next",
"noActionAssigned": "No action assigned",
"noActionAssignedForButton": "Could not perform {button}: No action assigned",
"noConnectionMethodIsConnectedOrActive": "No connection method is connected or active.",
"noConnectionMethodSelected": "No Connection Method selected",
"noControllerConnected": "None connected",
"noControllerUseCompanionMode": "No Controller? Use Companion Mode.",
"noIgnoredDevices": "No ignored devices.",
"noTrainerSelected": "No Trainer selected",
"notConnected": "Not connected",
"notificationDescription": "This keeps the app alive in background and updates you when the connection to your devices changes.",
"ok": "OK",
"openBikeControlActions": "OpenBikeControl actions",
"openBikeControlAnnouncement": "Great news - {trainerApp} supports the OpenBikeControl Protocol, so you'll have the best possible experience!",
"openBikeControlConnection": " e.g. by using OpenBikeControl connection",
"otherConnectionMethods": "Other Connection Methods",
"pairingDescription": "Pairing allows full customizability, but may not work on all devices.",
"pairingInstructions": "On your {targetName} go into Bluetooth settings and look for BikeControl or your machines name. Pairing is required if you want to use the remote control feature.",
"@pairingInstructions": {
"placeholders": {
"targetName": {
"type": "String"
}
}
},
"pairingInstructionsIOS": "On your iPad go to Settings > Accessibility > Touch > AssistiveTouch > Pointer Devices > Devices and pair your device. Make sure AssistiveTouch is enabled.",
"pasteExportedJsonData": "Paste the exported JSON data below:",
"pathCopiedToClipboard": "Path has been copied to clipboard",
"permissionsRequired": "In order for BikeControl to search for nearby devices and update you when the connection changes, please enable the following permissions:",
"platformNotSupported": "This {platform} is not supported :(",
"@platformNotSupported": {
"placeholders": {
"platform": {
"type": "String"
}
}
},
"platformRestrictionNotSupported": "Due to platform restrictions this scenario is not supported.",
"platformRestrictionOtherDevicesOnly": "Due to platform restrictions only controlling {appName} on other devices is supported.",
"@platformRestrictionOtherDevicesOnly": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"playPause": "Play/Pause",
"pleaseSelectAConnectionMethodFirst": "Please select a connection method in the Trainer settings, first.",
"predefinedAction": "Predefined {appName} action",
"@predefinedAction": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"pressButtonOnClickDevice": "Press a button on your Click device",
"pressKeyToAssign": "Press a key on your keyboard to assign to {buttonName}",
"@pressKeyToAssign": {
"placeholders": {
"buttonName": {
"type": "String"
}
}
},
"previous": "Previous",
"profileExportedToClipboard": "Profile \"{profileName}\" exported to clipboard",
"@profileExportedToClipboard": {
"placeholders": {
"profileName": {
"type": "String"
}
}
},
"profileImportedSuccessfully": "Profile imported successfully",
"profileName": "Profile name",
"purchase": "Purchase",
"recommendedConnectionMethods": "Recommended Connection Methods",
"removeFromIgnoredList": "Remove from ignored list",
"rename": "Rename",
"renameProfile": "Rename Profile",
"requirement": "Requirement",
"reset": "Reset",
"restart": "Restart",
"runAppOnPlatformRemotely": "Run {appName} on {platform} and control it remotely from this device{preferredConnection}.",
"@runAppOnPlatformRemotely": {
"placeholders": {
"appName": {
"type": "String"
},
"platform": {
"type": "String"
},
"preferredConnection": {
"type": "String"
}
}
},
"runAppOnThisDevice": "Run {appName} on this device.",
"@runAppOnThisDevice": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"save": "Save",
"scan": "SCAN",
"scanningForDevices": "Scanning for devices... Make sure they are powered on and in range and not connected to another device.",
"selectKeymap": "Select Keymap",
"selectTargetWhereAppRuns": "Select Target where {appName} runs on",
"@selectTargetWhereAppRuns": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"selectTrainerApp": "Select Trainer App",
"selectTrainerAppAndTarget": "Select Trainer App & Target Device",
"selectTrainerAppPlaceholder": "Select Trainer app",
"setting": "Setting",
"setupTrainer": "Setup Trainer",
"share": "Share",
"showDonation": "Show your appreciation by donating",
"showSupportedControllers": "Show Supported Controllers",
"showTroubleshootingGuide": "Show Troubleshooting Guide",
"signal": "Signal",
"simulateButtons": "Trainer Controls",
"simulateKeyboardShortcut": "Simulate Keyboard shortcut",
"simulateMediaKey": "Simulate Media key",
"simulateTouch": "Simulate Touch",
"stop": "Stop",
"targetOtherDevice": "Other Device",
"targetThisDevice": "This Device",
"theFollowingPermissionsRequired": "The following permissions are required:",
"touchAreaInstructions": "1. Create an in-game screenshot of your app (e.g. within MyWhoosh) in landscape orientation\n2. Load the screenshot with the button below\n3. The app is automatically set to landscape orientation for accurate mapping\n4. Press a button on your Click device to create a touch area\n5. Drag the touch areas to the desired position on the screenshot\n6. Save and close this screen",
"touchSimulationForegroundMessage": "To simulate touches the app needs to stay in the foreground.",
"trainer": "Trainer",
"trialDaysRemaining": "{trialDaysRemaining} days remaining",
"trialExpired": "Trial expired. Commands limited to {dailyCommandLimit} per day.",
"trialPeriodActive": "Trial Period Active - {trialDaysRemaining} days remaining",
"trialPeriodDescription": "Enjoy unlimited commands during your trial period. After the trial, commands will be limited to {dailyCommandLimit} per day.",
"troubleshootingGuide": "Troubleshooting Guide",
"tryingToConnectAgain": "Trying to connect again...",
"unassignAction": "Unassign action",
"unlockFullVersion": "Unlock Full Version",
"update": "Update",
"useCustomKeymapForButton": "Use a custom keymap to support the",
"version": "Version {version}",
"@version": {
"placeholders": {
"version": {
"type": "String"
}
}
},
"viewDetailedInstructions": "View Detailed Instructions",
"volumeDown": "Volume Down",
"volumeUp": "Volume Up",
"waiting": "Waiting...",
"waitingForConnectionKickrBike": "Waiting for connection. Choose KICKR BIKE PRO in {appName}'s controller pairing menu.",
"@waitingForConnectionKickrBike": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"whatsNew": "What's New",
"whyPermissionNeeded": "Why is this permission needed?",
"zwiftControllerAction": "Zwift Controller Action",
"zwiftControllerDescription": "Enables BikeControl to act as a Zwift-compatible controller."
}

427
lib/i10n/intl_fr.arb Normal file
View File

@@ -0,0 +1,427 @@
{
"accessibilityDescription": "BikeControl a besoin d'une autorisation d'accessibilité pour contrôler vos applications d'entraînement.",
"accessibilityDisclaimer": "BikeControl n'accédera à votre écran que pour effectuer les gestes que vous aurez configurés. Aucune autre fonctionnalité d'accessibilité ni aucune autre information personnelle ne sera accessible.",
"accessibilityReasonControl": "• Pour vous permettre de contrôler des applications telles que MyWhoosh, IndieVelo et d'autres à l'aide de vos appareils Zwift.",
"accessibilityReasonTouch": "• Pour simuler des gestes tactiles sur votre écran afin de contrôler les applications d'entraînement",
"accessibilityReasonWindow": "• Pour détecter quelle fenêtre de l'application d'entraînement est actuellement active",
"accessibilityServiceExplanation": "BikeControl doit utiliser l'API AccessibilityService d'Android pour fonctionner correctement.",
"accessibilityServiceNotRunning": "Le service d'accessibilité ne fonctionne pas.\nSuivez les instructions à l'adresse",
"accessibilityServicePermissionRequired": "Autorisation requise pour le service d'accessibilité",
"accessibilityUsageGestures": "• Lorsque vous appuyez sur les boutons de vos appareils Zwift Click, Zwift Ride ou Zwift Play, BikeControl simule des gestes tactiles à des emplacements spécifiques de l'écran.",
"accessibilityUsageMonitor": "• L'application surveille quelle fenêtre de l'application d'entraînement est active afin de s'assurer que les gestes sont envoyés à la bonne application.",
"accessibilityUsageNoData": "• Aucune donnée personnelle n'est consultée ou collectée par le biais de ce service.",
"accessories": "Accessoires",
"action": "Action",
"adjustControllerButtons": "Ajuster les boutons de la manette",
"allow": "Permettre",
"allowAccessibilityService": "Autoriser le service d'accessibilité",
"allowBluetoothConnections": "Autoriser les connexions Bluetooth",
"allowBluetoothScan": "Autoriser la recherche Bluetooth",
"allowLocationForBluetooth": "Autoriser la localisation pour que la recherche Bluetooth fonctionne",
"allowPersistentNotification": "Autoriser les notifications",
"allowsRunningInBackground": "Permet à BikeControl de continuer à fonctionner en arrière-plan",
"appIdActions": "{appId} actions",
"@appIdActions": {
"placeholders": {
"appId": {
"type": "String"
}
}
},
"battery": "Batterie",
"bluetoothAdvertiseAccess": "Accès à la publicité Bluetooth",
"bluetoothTurnedOn": "Bluetooth activé",
"browserNotSupported": "Ce navigateur ne prend pas en charge Web Bluetooth et la plateforme n'est pas prise en charge :(",
"button": "bouton.",
"cancel": "Annuler",
"changelog": "Journal des modifications",
"checkMyWhooshConnectionScreen": "Vérifiez l'écran de connexion dans MyWhoosh pour voir si « Link » est connecté.",
"chooseAnotherScreenshot": "Choisissez une autre capture d'écran",
"chooseBikeControlInConnectionScreen": "Sélectionnez BikeControl dans l'écran de connexion.",
"clickAButtonOnYourController": "Cliquez sur un bouton de votre manette pour modifier son action ou appuyez sur l'icône de modification.",
"clickV2EventInfo": "Votre Click V2 n'envoie peut-être plus les événements liés aux boutons. Vérifiez en appuyant sur quelques boutons et voyez s'ils apparaissent dans BikeControl.",
"clickV2Instructions": "Pour que ton Zwift Click V2 marche super bien, tu dois le connecter à l'appli Zwift une fois par jour.\nSi tu ne le fais pas, le Click V2 s'arrêtera de fonctionner au bout d'une minute.\n\n1. Ouvre l'appli Zwift.\n2. Connecte-toi (pas besoin d'abonnement) et ouvre l'écran de connexion des appareils.\n3. Connecte ton home trainer, puis connecte le Zwift Click V2.\n4. Ferme l'appli Zwift et reconnecte-toi dans BikeControl.",
"close": "Fermer",
"commandsRemainingToday": "{commandsRemainingToday}/{dailyCommandLimit} commandes restantes aujourd'hui",
"configuration": "Configuration",
"connectControllerToPreview": "Connectez un périphérique de contrôle pour prévisualiser et personnaliser la configuration des touches.",
"connectControllers": "Connectez les contrôleurs",
"connectDirectlyOverNetwork": "Connexion directe via le réseau",
"connectToTrainerApp": "Se connecter à l'application Trainer",
"connectUsingBluetooth": "Se connecter via Bluetooth",
"connectUsingMyWhooshLink": "Connectez-vous à l'aide de MyWhoosh « Link »",
"connected": "Connecté",
"connectedControllers": "Contrôleurs connectés",
"connectedTo": "Connecté à {appId}",
"@connectedTo": {
"placeholders": {
"appId": {
"type": "String"
}
}
},
"connection": "Connexion",
"continueAction": "Continuer",
"controlAppUsingModes": "Contrôle {appName} en utilisant{modes}",
"@controlAppUsingModes": {
"placeholders": {
"appName": {
"type": "String"
},
"modes": {
"type": "String"
}
}
},
"controllers": "Contrôleurs",
"couldNotPerformButtonnamesplitbyuppercaseNoKeymapSet": "Impossible d'effectuer {button}: Aucun clavier défini",
"create": "Créer",
"createNewKeymap": "Créer un nouveau clavier",
"createNewProfileByDuplicating": "Créer un nouveau profil personnalisé en dupliquant «{profileName} ».",
"@createNewProfileByDuplicating": {
"placeholders": {
"profileName": {
"type": "String"
}
}
},
"createdNewCustomProfile": "Création d'un nouveau profil personnalisé : {profileName}",
"@createdNewCustomProfile": {
"placeholders": {
"profileName": {
"type": "String"
}
}
},
"customizeControllerButtons": "Personnalisez les boutons de la manette pour {appName}",
"@customizeControllerButtons": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"customizeKeymapHint": "Personnalisez la configuration des touches si vous rencontrez des problèmes (par exemple, une sortie clavier incorrecte ou un placement des touches mal aligné).",
"dailyCommandLimitReachedNotification": "Limite de commandes journalières atteinte pour aujourd'hui. Passez à la version supérieure pour débloquer la version complète avec commandes illimitées.",
"dailyLimitReached": "Limite journalière atteinte ({dailyCommandCount} /{dailyCommandLimit} utilisé)",
"delete": "Supprimer",
"deleteProfile": "Supprimer le profil",
"deleteProfileConfirmation": "Êtes-vous sûr de vouloir supprimer «{profileName} » ? Cette action ne peut pas être annulée.",
"@deleteProfileConfirmation": {
"placeholders": {
"profileName": {
"type": "String"
}
}
},
"deny": "Refuser",
"deviceButton": "{deviceName} bouton",
"@deviceButton": {
"placeholders": {
"deviceName": {
"type": "String"
}
}
},
"disconnectDevices": "Déconnecter les appareils",
"disconnected": "Déconnecté",
"donateByBuyingFromPlayStore": "en achetant l'application sur le Play Store",
"donateViaCreditCard": "par carte bancaire, Google Pay, Apple Pay et autres",
"donateViaPaypal": "via PayPal",
"download": "Télécharger",
"dragToReposition": "Faites glisser pour repositionner",
"duplicate": "Double",
"enableAutoRotation": "Activez la rotation automatique sur votre appareil pour vous assurer que l'application fonctionne correctement.",
"enableBluetooth": "Activer le Bluetooth",
"enableKeyboardAccessMessage": "Activez l'accès au clavier dans l'écran suivant pour BikeControl. Si vous ne voyez pas BikeControl, veuillez l'ajouter manuellement.",
"enableKeyboardMouseControl": "Activez le contrôle du clavier et de la souris pour une meilleure interaction avec {appName} Une fois activé, aucune autre action ni connexion n'est nécessaire. BikeControl enverra directement les entrées de la souris ou du clavier à {appName} .",
"@enableKeyboardMouseControl": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"enableMediaKeyDetection": "Activer la détection des touches multimédias",
"enablePairingProcess": "Activer le processus d'appairage",
"enablePermissions": "Activer les autorisations",
"enableSteeringWithPhone": "Activez les capteurs du téléphone pour permettre, par exemple, la direction.",
"enableVibrationFeedback": "Activer le retour haptique par vibration lors du changement de vitesse",
"enableZwiftControllerBluetooth": "Activer le contrôleur Zwift (Bluetooth)",
"enableZwiftControllerNetwork": "Activer le contrôleur Zwift (réseau)",
"errorStartingMyWhooshLink": "Erreur lors du démarrage du serveur MyWhoosh Link. Veuillez vous assurer que l'application « MyWhoosh Link » n'est pas déjà en cours d'exécution sur cet appareil.",
"errorStartingOpenBikeControlBluetoothServer": "Erreur lors du démarrage du serveur Bluetooth OpenBikeControl.",
"errorStartingOpenBikeControlServer": "Erreur lors du démarrage du serveur OpenBikeControl.",
"exportAction": "Exporter",
"failedToImportProfile": "Échec de l'importation du profil. Format non valide.",
"failedToUpdate": "Échec de la mise à jour: {error}",
"@failedToUpdate": {
"placeholders": {
"error": {
"type": "String"
}
}
},
"firmware": "Firmware",
"forceCloseToUpdate": "Fermez de force l'application pour utiliser la nouvelle version.",
"fullVersion": "Version complète",
"fullVersionDescription": "La version complète comprend:\n- Commandes illimitées par jour\n- Accès à toutes les mises à jour futures \n- Aucun abonnement ! Un paiement unique :)",
"getSupport": "Obtenir de l'aide",
"gotIt": "Compris !",
"grant": "Accorder",
"granted": "Accordé",
"helpRequested": "Aide demandée pour BikeControl v{version}",
"@helpRequested": {
"placeholders": {
"version": {
"type": "String"
}
}
},
"howBikeControlUsesPermission": "Comment BikeControl utilise-t-il cette autorisation ?",
"ignoredDevices": "Appareils ignorés",
"importAction": "Importer",
"importProfile": "Profil d'importation",
"instructions": "Mode d'emploi",
"jsonData": "Données JSON",
"keyboardAccess": "Accès clavier",
"latestVersion": "dernier: {version}",
"@latestVersion": {
"placeholders": {
"version": {
"type": "String"
}
}
},
"leaveAReview": "Laisser un avis",
"letsAppConnectOverBluetooth": "Connectons {appName} à BikeControl via Bluetooth.",
"@letsAppConnectOverBluetooth": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"letsAppConnectOverNetwork": "Permet à {appName} de se connecter directement via le réseau. Sélectionnez BikeControl dans l'écran de connexion.",
"@letsAppConnectOverNetwork": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"license": "Licence",
"licenseStatus": "Statut de la licence",
"loadScreenshotForPlacement": "Charger une capture d'écran du jeu pour le placement",
"logViewer": "Visionneuse de journaux",
"logs": "Journaux",
"logsAreAlsoAt": "Les journaux sont également à",
"logsHaveBeenCopiedToClipboard": "Les journaux ont été copiés dans le presse-papiers.",
"longPress": "long\nappui",
"longPressMode": "Mode appui long (par rapport à la répétition)",
"mailSupportExplanation": "Répondre à tout le monde individuellement par e-mail, ça me prend beaucoup de temps.\n\nSi t'as des questions ou des problèmes, pense à utiliser Reddit, Facebook ou GitHub pour que tout le monde puisse en profiter.",
"manageIgnoredDevices": "Gérer les périphériques ignorés",
"manageProfile": "Gérer mon profil",
"manualyControllingButton": "Contrôle {trainerApp} manuellement!",
"mediaKeyDetectionTooltip": "Activez cette option pour permettre à BikeControl de détecter les télécommandes Bluetooth. Pour ce faire, BikeControl doit fonctionner comme un lecteur multimédia.",
"miuiDeviceDetected": "Appareil MIUI détecté",
"miuiDisableBatteryOptimization": "• Désactivez l'optimisation de la batterie pour BikeControl.",
"miuiEnableAutostart": "• Activer le démarrage automatique pour BikeControl",
"miuiEnsureProperWorking": "Pour garantir le bon fonctionnement de BikeControl :",
"miuiLockInRecentApps": "• Verrouiller l'application dans les applications récentes",
"miuiWarningDescription": "Votre appareil fonctionne sous MIUI, qui est connu pour supprimer de manière agressive les services d'arrière-plan et les services d'accessibilité.",
"moreInformation": "Plus d'informations",
"mustChooseAllowOrDeny": "Vous devez choisir d'autoriser ou de refuser cette autorisation pour continuer.",
"myWhooshDirectConnectAction": "Action «Link» de MyWhoosh",
"myWhooshDirectConnection": " par exemple en utilisant MyWhoosh «Link».",
"myWhooshLinkConnected": "MyWhoosh « Link » connecté",
"myWhooshLinkDescriptionLocal": "Connecte-toi directement à MyWhoosh avec la méthode « Link ». Tu peux faire des trucs comme changer de vitesse, utiliser des émoticônes, indiquer la direction à prendre, et plein d'autres choses. L'appli MyWhoosh Link ne doit PAS être ouverte en même temps.",
"myWhooshLinkInfo": "Si tu rencontres des problèmes, jette un œil à la section dépannage. Une méthode de connexion bien plus fiable sera bientôt disponible !",
"needHelpClickHelp": "Besoin d'aide ? Cliquez sur le",
"needHelpDontHesitate": "bouton en haut et n'hésitez pas à nous contacter.",
"newConnectionMethodAnnouncement": "{trainerApp} prendra bientôt en charge des méthodes de connexion bien meilleures et plus fiables - restez à l'écoute pour les mises à jour !",
"newCustomProfile": "Nouveau profil personnalisé",
"newProfileName": "Nouveau nom de profil",
"newVersionAvailable": "Nouvelle version disponible",
"newVersionAvailableWithVersion": "Nouvelle version disponible: {version}",
"@newVersionAvailableWithVersion": {
"placeholders": {
"version": {
"type": "String"
}
}
},
"next": "Suivant",
"noActionAssigned": "Aucune action assignée",
"noActionAssignedForButton": "Impossible d'effectuer {button}: Aucune action assignée",
"noConnectionMethodIsConnectedOrActive": "Aucune méthode de connexion n'est établie ou active.",
"noConnectionMethodSelected": "Aucune méthode de connexion choisie",
"noControllerConnected": "Aucun connecté",
"noControllerUseCompanionMode": "Pas de manette ? Utilisez le mode compagnon.",
"noIgnoredDevices": "Aucun appareil ignoré.",
"noTrainerSelected": "Aucun Trainer sélectionné",
"notConnected": "Non connecté",
"notificationDescription": "Cela permet à l'application de rester active en arrière-plan et de vous informer lorsque la connexion à vos appareils change.",
"ok": "OK",
"openBikeControlActions": "Actions OpenBikeControl",
"openBikeControlAnnouncement": "Excellente nouvelle! {trainerApp} Il prend en charge le protocole OpenBikeControl, vous bénéficierez donc de la meilleure expérience possible !",
"openBikeControlConnection": " par exemple en utilisant la connexion OpenBikeControl",
"otherConnectionMethods": "Autres méthodes de connexion",
"pairingDescription": "Le jumelage permet une personnalisation complète, mais peut ne pas fonctionner sur tous les appareils.",
"pairingInstructions": "Sur votre {targetName}, accédez aux paramètres Bluetooth et recherchez BikeControl ou le nom de votre appareil. L'appairage est nécessaire si vous souhaitez utiliser la fonction de commande à distance.",
"@pairingInstructions": {
"placeholders": {
"targetName": {
"type": "String"
}
}
},
"pairingInstructionsIOS": "Sur votre iPad, allez dans Réglages > Accessibilité > Toucher > AssistiveTouch > Périphériques de pointage > Périphériques et appairez votre appareil. Assurez-vous que la fonction AssistiveTouch est activée.",
"pasteExportedJsonData": "Collez ci-dessous les données JSON exportées :",
"pathCopiedToClipboard": "Le chemin a été copié dans le presse-papiers.",
"permissionsRequired": "Pour que BikeControl puisse rechercher les appareils à proximité et vous informer en cas de changement de connexion, veuillez activer les autorisations suivantes:",
"platformNotSupported": "Cette {platform} n'est pas prise en charge :(",
"@platformNotSupported": {
"placeholders": {
"platform": {
"type": "String"
}
}
},
"platformRestrictionNotSupported": "En raison des restrictions de la plateforme, ce scénario n'est pas pris en charge.",
"platformRestrictionOtherDevicesOnly": "En raison des restrictions de la plateforme, seul le contrôle de l'{appName} s sur d'autres appareils est pris en charge.",
"@platformRestrictionOtherDevicesOnly": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"playPause": "Lecture/Pause",
"pleaseSelectAConnectionMethodFirst": "Commence par choisir une méthode de connexion dans les paramètres du Trainer.",
"predefinedAction": "Action prédéfinie d'{appName}",
"@predefinedAction": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"pressButtonOnClickDevice": "Appuyez sur un bouton de votre appareil Click",
"pressKeyToAssign": "Appuyez sur une touche de votre clavier pour l'attribuer à {buttonName}",
"@pressKeyToAssign": {
"placeholders": {
"buttonName": {
"type": "String"
}
}
},
"previous": "Précédent",
"profileExportedToClipboard": "Profil «{profileName} » exporté vers le presse-papiers",
"@profileExportedToClipboard": {
"placeholders": {
"profileName": {
"type": "String"
}
}
},
"profileImportedSuccessfully": "Profil importé avec succès",
"profileName": "Nom du profil",
"purchase": "Achat",
"recommendedConnectionMethods": "Méthodes de connexion recommandées",
"removeFromIgnoredList": "Supprimer de la liste ignorée",
"rename": "Rebaptiser",
"renameProfile": "Renommer le profil",
"requirement": "Exigence",
"reset": "Réinitialiser",
"restart": "Redémarrage",
"runAppOnPlatformRemotely": "Exécutez {appName} sur {platform} et contrôlez-le à distance depuis cet appareil{preferredConnection}.",
"@runAppOnPlatformRemotely": {
"placeholders": {
"appName": {
"type": "String"
},
"platform": {
"type": "String"
},
"preferredConnection": {
"type": "String"
}
}
},
"runAppOnThisDevice": "Exécutez {appName} sur cet appareil.",
"@runAppOnThisDevice": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"save": "Sauvegarder",
"scan": "BALAYAGE",
"scanningForDevices": "Recherche d'appareils en cours... Assurez-vous qu'ils sont allumés, à portée et non connectés à un autre appareil.",
"selectKeymap": "Sélectionner le clavier",
"selectTargetWhereAppRuns": "Sélectionnez la cible où {appName} fonctionne sur",
"@selectTargetWhereAppRuns": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"selectTrainerApp": "Sélectionner l'application Trainer",
"selectTrainerAppAndTarget": "Sélectionnez l'application Trainer et l'appareil cible",
"selectTrainerAppPlaceholder": "Sélectionner l'application Trainer",
"setting": "Paramètre",
"setupTrainer": "Configurer le Trainer",
"share": "Partager",
"showDonation": "Exprimez votre reconnaissance en faisant un don",
"showSupportedControllers": "Afficher les manettes compatibles",
"showTroubleshootingGuide": "Afficher le guide de dépannage",
"signal": "Signal",
"simulateButtons": "Commandes de l'entraîneur",
"simulateKeyboardShortcut": "Simuler un raccourci clavier",
"simulateMediaKey": "Simuler la touche média",
"simulateTouch": "Simuler le toucher",
"stop": "Arrêt",
"targetOtherDevice": "Autre appareil",
"targetThisDevice": "Cet appareil",
"theFollowingPermissionsRequired": "Les autorisations suivantes sont requises :",
"touchAreaInstructions": "1. Créez une capture d'écran de votre application (par exemple dans MyWhoosh) en mode paysage.\n2. Chargez la capture d'écran à l'aide du bouton ci-dessous.\n3. L'application est automatiquement réglée en mode paysage pour un mappage précis.\n4. Appuyez sur un bouton de votre appareil Click pour créer une zone tactile.\n5. Faites glisser les zones tactiles vers la position souhaitée sur la capture d'écran.\n6. Enregistrez et fermez cet écran.",
"touchSimulationForegroundMessage": "Pour simuler les touches, l'application doit rester au premier plan.",
"trainer": "Trainer",
"trialDaysRemaining": "{trialDaysRemaining} jours restants",
"trialExpired": "Période d'essai expirée. Commandes limitées à {dailyCommandLimit} par jour.",
"trialPeriodActive": "Période d'essai active -{trialDaysRemaining} jours restants",
"trialPeriodDescription": "Profitez de commandes illimitées pendant votre période d'essai. Après la période d'essai, le nombre de commandes sera limité à {dailyCommandLimit} par jour.",
"troubleshootingGuide": "Guide de dépannage",
"tryingToConnectAgain": "Tentative de reconnexion...",
"unassignAction": "Action de désaffectation",
"unlockFullVersion": "Débloquer la version complète",
"update": "Mise à jour",
"useCustomKeymapForButton": "Utilisez une configuration de touches personnalisée pour prendre en charge",
"version": "Version {version}",
"@version": {
"placeholders": {
"version": {
"type": "String"
}
}
},
"viewDetailedInstructions": "Consultez les instructions détaillées",
"volumeDown": "Baisser le volume",
"volumeUp": "Augmenter le volume",
"waiting": "En attendant...",
"waitingForConnectionKickrBike": "En attente de connexion. Sélectionnez KICKR BIKE PRO dans le menu de couplage du contrôleur de l'{appName}.",
"@waitingForConnectionKickrBike": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"whatsNew": "Quoi de neuf",
"whyPermissionNeeded": "Pourquoi cette autorisation est-elle nécessaire ?",
"zwiftControllerAction": "Action du contrôleur Zwift",
"zwiftControllerDescription": "Permet à BikeControl de fonctionner comme une manette compatible avec Zwift."
}

427
lib/i10n/intl_pl.arb Normal file
View File

@@ -0,0 +1,427 @@
{
"accessibilityDescription": "BikeControl potrzebuje uprawnień dostępu, aby móc sterować aplikacjami treningowymi.",
"accessibilityDisclaimer": "BikeControl będzie miał dostęp do Twojego ekranu wyłącznie w celu wykonania skonfigurowanych przez Ciebie gestów. Nie będzie miał dostępu do żadnych innych funkcji ułatwień dostępu ani danych osobowych.",
"accessibilityReasonControl": "• Aby umożliwić Ci sterowanie aplikacjami takimi jak MyWhoosh, IndieVelo i innymi za pomocą urządzeń Zwift",
"accessibilityReasonTouch": "• Aby symulować gesty dotykowe na ekranie w celu sterowania aplikacją treningową",
"accessibilityReasonWindow": "• Aby wykryć, które okno aplikacji treningowej jest aktualnie aktywne",
"accessibilityServiceExplanation": "Aby działać prawidłowo, BikeControl musi korzystać z interfejsu API AccessibilityService systemu Android.",
"accessibilityServiceNotRunning": "Usługa ułatwień dostępu nie działa.\nPostępuj zgodnie z instrukcjami na",
"accessibilityServicePermissionRequired": "Wymagane uprawnienie usługi ułatwień dostępu",
"accessibilityUsageGestures": "• Gdy naciskasz przyciski na urządzeniach Zwift Click, Zwift Ride lub Zwift Play, BikeControl symuluje gesty dotykowe w określonych miejscach ekranu",
"accessibilityUsageMonitor": "• Aplikacja monitoruje, które okno aplikacji treningowej jest aktywne, aby zapewnić, że gesty są wysyłane do właściwej aplikacji",
"accessibilityUsageNoData": "• Ta usługa nie uzyskuje dostępu do danych osobowych, ani ich nie gromadzi",
"accessories": "Akcesoria",
"action": "Działanie",
"adjustControllerButtons": "Dostosuj przyciski kontrolera",
"allow": "Zezwól",
"allowAccessibilityService": "Zezwól na usługę ułatwień dostępu",
"allowBluetoothConnections": "Zezwól na połączenia Bluetooth",
"allowBluetoothScan": "Zezwól na skanowanie Bluetooth",
"allowLocationForBluetooth": "Zezwól na dostęp do lokalizacji, aby umożliwić skanowanie Bluetooth",
"allowPersistentNotification": "Zezwól na powiadomienia",
"allowsRunningInBackground": "Umożliwia działanie BikeControl w tle",
"appIdActions": "{appId} działania",
"@appIdActions": {
"placeholders": {
"appId": {
"type": "String"
}
}
},
"battery": "Bateria",
"bluetoothAdvertiseAccess": "Dostęp do reklamy Bluetooth",
"bluetoothTurnedOn": "Włączono Bluetooth",
"browserNotSupported": "Ta przeglądarka nie obsługuje technologii Web Bluetooth i platforma nie jest obsługiwana :(",
"button": "przycisk.",
"cancel": "Anuluj",
"changelog": "Changelog",
"checkMyWhooshConnectionScreen": "Sprawdź ekran połączenia w MyWhoosh, aby zobaczyć, czy „Link” jest połączony.",
"chooseAnotherScreenshot": "Wybierz inny zrzut ekranu",
"chooseBikeControlInConnectionScreen": "Wybierz BikeControl na ekranie połączenia.",
"clickAButtonOnYourController": "Kliknij przycisk na kontrolerze, aby edytować jego akcję lub naciśnij ikonę edycji.",
"clickV2EventInfo": "Twóje Click V2 może już nie wysyłać zdarzeń przycisków. Sprawdź, naciskając kilka przycisków, czy są one widoczne w BikeControl.",
"clickV2Instructions": "Aby Twoje Zwift Click V2 działały optymalnie, należy połączyć je z aplikacją Zwift przed każdym treningiem.\nJeśli tego nie zrobisz, Click V2 przestanie działać po minucie.\n\n1. Otwórz aplikację Zwift\n2. Zaloguj się (nie wymaga subskrypcji) i otwórz ekran połączeń z urządzeniami\n3. Połącz się z trenażerem, a następnie z Zwift Click V2\n4. Zamknij aplikację Zwift i ponownie połącz się z BikeControl",
"close": "Zamknij",
"commandsRemainingToday": "{commandsRemainingToday}/{dailyCommandLimit} dzisiejsze pozostałe polecenia",
"configuration": "Konfiguracja",
"connectControllerToPreview": "Podłącz urządzenie sterujące, aby wyświetlić podgląd i dostosować mapę klawiszy.",
"connectControllers": "Połącz kontrolery",
"connectDirectlyOverNetwork": "Połącz bezpośrednio przez sieć",
"connectToTrainerApp": "Połącz z aplikacją treningową",
"connectUsingBluetooth": "Połącz przez Bluetooth",
"connectUsingMyWhooshLink": "Połącz się za pomocą MyWhoosh „Link”",
"connected": "Połączono",
"connectedControllers": "Połączone kontrolery",
"connectedTo": "Połączono z {appId}",
"@connectedTo": {
"placeholders": {
"appId": {
"type": "String"
}
}
},
"connection": "Połączenie",
"continueAction": "Kontynuuj",
"controlAppUsingModes": "Kontroluj {appName} za pomocą {modes}",
"@controlAppUsingModes": {
"placeholders": {
"appName": {
"type": "String"
},
"modes": {
"type": "String"
}
}
},
"controllers": "Kontrolery",
"couldNotPerformButtonnamesplitbyuppercaseNoKeymapSet": "Nie można wykonać {button}: Nie zmapowano klawisza",
"create": "Utwórz",
"createNewKeymap": "Utwórz nową mapę klawiszy",
"createNewProfileByDuplicating": "Utwórz nowy profil niestandardowy, kopiując „{profileName}\"",
"@createNewProfileByDuplicating": {
"placeholders": {
"profileName": {
"type": "String"
}
}
},
"createdNewCustomProfile": "Utworzono nowy profil niestandardowy: {profileName}",
"@createdNewCustomProfile": {
"placeholders": {
"profileName": {
"type": "String"
}
}
},
"customizeControllerButtons": "Dostosuj przyciski kontrolera dla {appName}",
"@customizeControllerButtons": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"customizeKeymapHint": "Jeśli występują jakiekolwiek problemy (np. nieprawidłowe działanie klawiatury lub nieprawidłowe rozmieszczenie przycisków dotykowych), dostosuj mapę klawiszy.",
"dailyCommandLimitReachedNotification": "Dzienny limit poleceń został osiągnięty. Zaktualizuj do pełnej wersji z nieograniczoną liczbą poleceń.",
"dailyLimitReached": "Osiągnięto dzienny limit ({dailyCommandCount} /{dailyCommandLimit} wykorzystanych)",
"delete": "Usuń",
"deleteProfile": "Usuń profil",
"deleteProfileConfirmation": "Czy na pewno chcesz usunąć „{profileName}\"? Tej czynności nie można cofnąć.",
"@deleteProfileConfirmation": {
"placeholders": {
"profileName": {
"type": "String"
}
}
},
"deny": "Odmów",
"deviceButton": "{deviceName} przycisk",
"@deviceButton": {
"placeholders": {
"deviceName": {
"type": "String"
}
}
},
"disconnectDevices": "Odłącz urządzenia",
"disconnected": "Rozłączony",
"donateByBuyingFromPlayStore": "kupując aplikację w Play Store",
"donateViaCreditCard": "za pomocą karty kredytowej, Google Pay, Apple Pay i innych",
"donateViaPaypal": "przez PayPal",
"download": "Pobierz",
"dragToReposition": "Przeciągnij, aby zmienić położenie",
"duplicate": "Duplikuj",
"enableAutoRotation": "Włącz funkcję automatycznego obracania na urządzeniu, aby mieć pewność, że aplikacja będzie działać prawidłowo.",
"enableBluetooth": "Włącz Bluetooth",
"enableKeyboardAccessMessage": "Włącz dostęp do klawiatury dla BikeControl na poniższym ekranie. Jeśli nie widzisz BikeControl, dodaj go ręcznie.",
"enableKeyboardMouseControl": "Włącz sterowanie klawiaturą i myszą, aby zapewnić lepszą interakcję z {appName}. Po aktywacji nie jest wymagane żadne dalsze działanie, ani połączenie. BikeControl będzie bezpośrednio wysyłał dane z myszy lub klawiatury do {appName}.",
"@enableKeyboardMouseControl": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"enableMediaKeyDetection": "Włącz rozpoznawanie klawiszy multimedialnych",
"enablePairingProcess": "Włącz proces parowania",
"enablePermissions": "Nadaj uprawnienia",
"enableSteeringWithPhone": "Włącz czujniki telefonów, aby umożliwić np. sterowanie",
"enableVibrationFeedback": "Włącz wibracje podczas zmiany biegów",
"enableZwiftControllerBluetooth": "Włącz kontroler Zwift (Bluetooth)",
"enableZwiftControllerNetwork": "Włącz kontroler Zwift (sieć)",
"errorStartingMyWhooshLink": "Błąd podczas uruchamiania serwera MyWhoosh Link. Upewnij się, że aplikacja „MyWhoosh Link” nie jest już uruchomiona na tym urządzeniu.",
"errorStartingOpenBikeControlBluetoothServer": "Błąd podczas uruchamiania serwera Bluetooth OpenBikeControl.",
"errorStartingOpenBikeControlServer": "Błąd podczas uruchamiania serwera OpenBikeControl.",
"exportAction": "Eksport",
"failedToImportProfile": "Nie udało się zaimportować profilu. Nieprawidłowy format.",
"failedToUpdate": "Nie udało się zaktualizować: {error}",
"@failedToUpdate": {
"placeholders": {
"error": {
"type": "String"
}
}
},
"firmware": "Firmware",
"forceCloseToUpdate": "Wymuś zamknięcie aplikacji, aby móc korzystać z nowej wersji",
"fullVersion": "Pełna wersja",
"fullVersionDescription": "Pełna wersja zawiera: \n- Nielimitowane polecenia dziennie \n- Dostęp do wszystkich przyszłych aktualizacji \n- Brak subskrypcji! Opłata jednorazowa :)",
"getSupport": "Uzyskaj wsparcie",
"gotIt": "Zrozumiałem!",
"grant": "Nadaj",
"granted": "Nadano",
"helpRequested": "Prośba o pomoc dotyczącą BikeControl v{version}",
"@helpRequested": {
"placeholders": {
"version": {
"type": "String"
}
}
},
"howBikeControlUsesPermission": "W jaki sposób BikeControl wykorzystuje to uprawnienie?",
"ignoredDevices": "Zignorowane urządzenia",
"importAction": "Import",
"importProfile": "Importuj profil",
"instructions": "Instrukcje",
"jsonData": "Dane JSON",
"keyboardAccess": "Dostęp do klawiatury",
"latestVersion": "najnowszy: {version}",
"@latestVersion": {
"placeholders": {
"version": {
"type": "String"
}
}
},
"leaveAReview": "Zostaw recenzję",
"letsAppConnectOverBluetooth": "Pozwala {appName} połączyć się z BikeControl przez Bluetooth.",
"@letsAppConnectOverBluetooth": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"letsAppConnectOverNetwork": "Pozwala {appName} połączyć się bezpośrednio przez sieć. Wybierz BikeControl na ekranie połączenia.",
"@letsAppConnectOverNetwork": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"license": "Licencja",
"licenseStatus": "Status licencji",
"loadScreenshotForPlacement": "Prześlij zrzut ekranu z gry w celu ustalenia miejsca",
"logViewer": "Przeglądarka dziennika zdarzeń",
"logs": "Dziennik zdarzeń",
"logsAreAlsoAt": "Dziennik zdarzeń jest dostępny również na",
"logsHaveBeenCopiedToClipboard": "Dziennik zdarzeń został skopiowany do schowka",
"longPress": "długie\nnaciśnięcie",
"longPressMode": "Tryb długiego naciśnięcia (zamiast powtarzania)",
"mailSupportExplanation": "Udzielanie indywidualnego wsparcia przez e-mail to dla mnie dużo pracy.\n\nProszę rozważyć korzystanie z Reddita, Facebooka lub GitHuba w celu zadawania pytań i zgłaszania problemów, aby cała społeczność mogła z tego skorzystać.",
"manageIgnoredDevices": "Zarządzaj ignorowanymi urządzeniami",
"manageProfile": "Zarządzaj profilem",
"manualyControllingButton": "Steruj {trainerApp} manualnie!",
"mediaKeyDetectionTooltip": "Włącz tę opcję, aby umożliwić BikeControl wykrywanie pilotów Bluetooth.\nW tym celu BikeControl musi działać jako odtwarzacz multimedialny.",
"miuiDeviceDetected": "Wykryto urządzenie MIUI",
"miuiDisableBatteryOptimization": "• Wyłącz optymalizację baterii dla BikeControl",
"miuiEnableAutostart": "• Włącz autostart dla BikeControl",
"miuiEnsureProperWorking": "Aby mieć pewność, że BikeControl działa prawidłowo:",
"miuiLockInRecentApps": "• Zablokuj aplikację w ostatnio używanych aplikacjach",
"miuiWarningDescription": "Na Twoim urządzeniu działa oprogramowanie MIUI, który słynie z agresywnego wyłączania usług działających w tle i usług ułatwień dostępu.",
"moreInformation": "Więcej informacji",
"mustChooseAllowOrDeny": "Aby kontynuować, musisz zezwolić lub odmówić tego uprawnienia.",
"myWhooshDirectConnectAction": "Akcja „Link” MyWhoosh",
"myWhooshDirectConnection": " np. za pomocą MyWhoosh „Link”",
"myWhooshLinkConnected": "Połączono MyWhoosh „Link”",
"myWhooshLinkDescriptionLocal": "Połącz się bezpośrednio z MyWhoosh za pomocą „Link”. Obsługiwane akcje to m.in. zmiana biegów, emotikony i zmiana kierunku. Aplikacja MyWhoosh Link NIE może być uruchomiona w tym samym czasie.",
"myWhooshLinkInfo": "W razie napotkania błędów prosimy o sprawdzenie sekcji rozwiązywania problemów. Wkrótce pojawi się znacznie bardziej niezawodna metoda połączenia!",
"needHelpClickHelp": "Potrzebujesz pomocy? Kliknij",
"needHelpDontHesitate": "przycisk na górze i prosimy o kontakt.",
"newConnectionMethodAnnouncement": "{trainerApp} już wkrótce będziemy wspierać znacznie lepsze i bardziej niezawodne metody połączeń — bądźcie na bieżąco z aktualizacjami!",
"newCustomProfile": "Nowy profil niestandardowy",
"newProfileName": "Nowa nazwa profilu",
"newVersionAvailable": "Dostępna jest nowa wersja",
"newVersionAvailableWithVersion": "Dostępna jest nowa wersja: {version}",
"@newVersionAvailableWithVersion": {
"placeholders": {
"version": {
"type": "String"
}
}
},
"next": "Dalej",
"noActionAssigned": "Brak przypisanej akcji",
"noActionAssignedForButton": "Nie można wykonać {button}: Brak przypisanej akcji",
"noConnectionMethodIsConnectedOrActive": "Żadna metoda połączenia nie jest podłączona lub aktywna.",
"noConnectionMethodSelected": "Nie wybrano metody połączenia",
"noControllerConnected": "Brak połączenia",
"noControllerUseCompanionMode": "Nie masz kontrolera? Użyj Companion Mode.",
"noIgnoredDevices": "Brak ignorowanych urządzeń.",
"noTrainerSelected": "Nie wybrano aplikacji treningowej",
"notConnected": "Nie połączono",
"notificationDescription": "Dzięki temu aplikacja działa w tle i powiadamia Cię o każdej zmianie połączenia z Twoim urządzeniem.",
"ok": "OK",
"openBikeControlActions": "Akcje OpenBikeControl",
"openBikeControlAnnouncement": "Świetna wiadomość - {trainerApp} obsługuje protokół OpenBikeControl, dzięki czemu uzyskasz najlepsze doświadczenia!",
"openBikeControlConnection": " np. za pomocą połączenia OpenBikeControl",
"otherConnectionMethods": "Inne metody połączenia",
"pairingDescription": "Parowanie umożliwia pełną personalizację, jednak może nie działać na wszystkich urządzeniach.",
"pairingInstructions": "Przejdź do ustawień Bluetooth na twoim {targetName} i wyszukaj BikeControl lub nazwę swojego urządzenia. Parowanie jest wymagane, jeśli chcesz korzystać z funkcji zdalnego sterowania.",
"@pairingInstructions": {
"placeholders": {
"targetName": {
"type": "String"
}
}
},
"pairingInstructionsIOS": "Na iPadzie przejdź do Ustawienia > Ułatwienia dostępu > Dotyk > AssistiveTouch > Urządzenia wskazujące > Urządzenia i sparuj twoje urządzenie. Upewnij się, że funkcja AssistiveTouch jest włączona.",
"pasteExportedJsonData": "Wklej wyeksportowane dane JSON poniżej:",
"pathCopiedToClipboard": "Ścieżka została skopiowana do schowka",
"permissionsRequired": "Aby aplikacja BikeControl mogła wyszukiwać urządzenia w pobliżu i informować o zmianie połączenia, włącz następujące uprawnienia:",
"platformNotSupported": "Platforma {platform} nie jest obsługiwana :(",
"@platformNotSupported": {
"placeholders": {
"platform": {
"type": "String"
}
}
},
"platformRestrictionNotSupported": "Ze względu na ograniczenia platformy ten scenariusz nie jest obsługiwany.",
"platformRestrictionOtherDevicesOnly": "Ze względu na ograniczenia platformy obsługiwane jest tylko sterowanie {appName} na innych urządzeniach.",
"@platformRestrictionOtherDevicesOnly": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"playPause": "Start/Pauza",
"pleaseSelectAConnectionMethodFirst": "Najpierw wybierz metodę połączenia w ustawieniach aplikacji treningowej.",
"predefinedAction": "Predefiniowana akcja {appName}",
"@predefinedAction": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"pressButtonOnClickDevice": "Wciśnij przycisk na swoim urządzeniu Click",
"pressKeyToAssign": "Naciśnij klawisz na klawiaturze, aby go przypisać do {buttonName}",
"@pressKeyToAssign": {
"placeholders": {
"buttonName": {
"type": "String"
}
}
},
"previous": "Poprzedni",
"profileExportedToClipboard": "Wyeksportowano profil „{profileName}\" do schowka",
"@profileExportedToClipboard": {
"placeholders": {
"profileName": {
"type": "String"
}
}
},
"profileImportedSuccessfully": "Pomyślnie zaimportowano profil",
"profileName": "Nazwa profilu",
"purchase": "Zakup",
"recommendedConnectionMethods": "Zalecane metody połączenia",
"removeFromIgnoredList": "Usuń z listy ignorowanych",
"rename": "Zmiana nazwy",
"renameProfile": "Zmień nazwę profilu",
"requirement": "Wymóg",
"reset": "Reset",
"restart": "Uruchom ponownie",
"runAppOnPlatformRemotely": "Uruchom {appName} na {platform} i steruj nim zdalnie z tego urządzenia {preferredConnection}.",
"@runAppOnPlatformRemotely": {
"placeholders": {
"appName": {
"type": "String"
},
"platform": {
"type": "String"
},
"preferredConnection": {
"type": "String"
}
}
},
"runAppOnThisDevice": "Uruchom {appName} na tym urządzeniu.",
"@runAppOnThisDevice": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"save": "Zapisz",
"scan": "SKANUJ",
"scanningForDevices": "Skanowanie urządzeń... Upewnij się, że są włączone, znajdują się w zasięgu i nie są podłączone do innego urządzenia.",
"selectKeymap": "Wybierz mapę klawiszy",
"selectTargetWhereAppRuns": "Wybierz urządzenie docelowe, na którym działa {appName}",
"@selectTargetWhereAppRuns": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"selectTrainerApp": "Wybierz aplikację treningową",
"selectTrainerAppAndTarget": "Wybierz aplikację treningową i docelowe urządzenie",
"selectTrainerAppPlaceholder": "Wybierz aplikację treningową",
"setting": "Ustawienie",
"setupTrainer": "Konfiguracja trenażera",
"share": "Udostępnij",
"showDonation": "Wyraź swoją wdzięczność poprzez darowiznę",
"showSupportedControllers": "Pokaż wspierane kontrolery",
"showTroubleshootingGuide": "Pokaż instrukcję rozwiązywania problemów",
"signal": "Sygnał",
"simulateButtons": "Sterowanie trenażerem",
"simulateKeyboardShortcut": "Symuluj skrót klawiaturowy",
"simulateMediaKey": "Symuluj przycisk multimedialny",
"simulateTouch": "Symuluj dotyk",
"stop": "Stop",
"targetOtherDevice": "Inne urządzenie",
"targetThisDevice": "To urządzenie",
"theFollowingPermissionsRequired": "Wymagane są następujące uprawnienia:",
"touchAreaInstructions": "1. Utwórz zrzut ekranu w grze swojej aplikacji (np. w MyWhoosh) w orientacji poziomej\n2. Załaduj zrzut ekranu za pomocą poniższego przycisku\n3. Aplikacja automatycznie ustawi się w orientacji poziomej, aby zapewnić dokładne odwzorowanie\n4. Naciśnij przycisk na urządzeniu Click, aby utworzyć obszar dotykowy\n5. Przeciągnij obszary dotykowe do żądanej pozycji na zrzucie ekranu\n6. Zapisz i zamknij ten ekran.",
"touchSimulationForegroundMessage": "Aby symulować dotyk, aplikacja musi pozostać na pierwszym planie.",
"trainer": "Trenażer",
"trialDaysRemaining": "Pozostało {trialDaysRemaining} dni",
"trialExpired": "Wersja próbna wygasła. Liczba poleceń ograniczona do {dailyCommandLimit} na dzień.",
"trialPeriodActive": "Okres próbny jest aktywny - pozostało {trialDaysRemaining} dni",
"trialPeriodDescription": "Korzystaj z nieograniczonej liczby poleceń w okresie próbnym. Po zakończeniu okresu próbnego polecenia będą ograniczone do {dailyCommandLimit} na dzień.",
"troubleshootingGuide": "Instrukcja rozwiązywania problemów",
"tryingToConnectAgain": "Próba ponownego połączenia...",
"unassignAction": "Anuluj przypisanie akcji",
"unlockFullVersion": "Odblokuj pełną wersję",
"update": "Aktualizacja",
"useCustomKeymapForButton": "Użyj niestandardowej mapy klawiszy, aby obsługiwać",
"version": "Wersja {version}",
"@version": {
"placeholders": {
"version": {
"type": "String"
}
}
},
"viewDetailedInstructions": "Zobacz szczegółowe instrukcje",
"volumeDown": "Zmniejsz głośność",
"volumeUp": "Zwiększ głośność",
"waiting": "Czekam...",
"waitingForConnectionKickrBike": "Czekam na połączenie. Wybierz KICKR BIKE PRO w menu parowania kontrolera w {appName}.",
"@waitingForConnectionKickrBike": {
"placeholders": {
"appName": {
"type": "String"
}
}
},
"whatsNew": "Co nowego",
"whyPermissionNeeded": "Dlaczego to uprawnienie jest potrzebne?",
"zwiftControllerAction": "Akcja kontrolera Zwift",
"zwiftControllerDescription": "Umożliwia BikeControl działanie jako kontroler kompatybilny ze Zwift."
}

View File

@@ -1,31 +1,137 @@
import 'dart:async';
import 'dart:io';
import 'dart:isolate';
import 'package:bike_control/bluetooth/messages/notification.dart';
import 'package:bike_control/gen/l10n.dart';
import 'package:bike_control/utils/actions/android.dart';
import 'package:bike_control/utils/actions/desktop.dart';
import 'package:bike_control/utils/actions/remote.dart';
import 'package:bike_control/widgets/menu.dart';
import 'package:bike_control/widgets/testbed.dart';
import 'package:bike_control/widgets/ui/colors.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter_local_notifications/flutter_local_notifications.dart';
import 'package:swift_control/pages/requirements.dart';
import 'package:swift_control/theme.dart';
import 'package:swift_control/utils/actions/android.dart';
import 'package:swift_control/utils/actions/desktop.dart';
import 'package:swift_control/utils/actions/remote.dart';
import 'package:swift_control/utils/settings/settings.dart';
import 'package:flutter_localizations/flutter_localizations.dart'
show GlobalMaterialLocalizations, GlobalWidgetsLocalizations;
import 'package:shadcn_flutter/shadcn_flutter.dart';
import 'bluetooth/connection.dart';
import 'bluetooth/devices/link/link.dart';
import 'pages/navigation.dart';
import 'utils/actions/base_actions.dart';
import 'utils/core.dart';
final connection = Connection();
final navigatorKey = GlobalKey<NavigatorState>();
late BaseActions actionHandler;
final flutterLocalNotificationsPlugin = FlutterLocalNotificationsPlugin();
final settings = Settings();
final whooshLink = WhooshLink();
const screenshotMode = false;
var screenshotMode = false;
void main() async {
WidgetsFlutterBinding.ensureInitialized();
// setup crash reporting
runApp(const SwiftPlayApp());
// Catch errors that happen in other isolates
if (!kIsWeb) {
Isolate.current.addErrorListener(
RawReceivePort((dynamic pair) {
final List<dynamic> errorAndStack = pair as List<dynamic>;
final error = errorAndStack.first;
final stack = errorAndStack.last as StackTrace?;
recordError(error, stack, context: 'Isolate');
}).sendPort,
);
}
runZonedGuarded<Future<void>>(
() async {
// Catch Flutter framework errors (build/layout/paint)
FlutterError.onError = (FlutterErrorDetails details) {
_recordFlutterError(details);
// Optionally forward to default behavior in debug:
FlutterError.presentError(details);
};
// Catch errors from platform dispatcher (async)
PlatformDispatcher.instance.onError = (Object error, StackTrace stack) {
recordError(error, stack, context: 'PlatformDispatcher');
// Return true means "handled"
return true;
};
WidgetsFlutterBinding.ensureInitialized();
final error = await core.settings.init();
runApp(BikeControlApp(error: error));
},
(Object error, StackTrace stack) {
if (kDebugMode) {
print('App crashed: $error');
debugPrintStack(stackTrace: stack);
}
recordError(error, stack, context: 'Zone');
},
);
}
Future<void> _recordFlutterError(FlutterErrorDetails details) async {
await _persistCrash(
type: 'flutter',
error: details.exceptionAsString(),
stack: details.stack,
information: details.informationCollector?.call().join('\n'),
);
}
Future<void> recordError(
Object error,
StackTrace? stack, {
required String context,
}) async {
await _persistCrash(
type: 'dart',
error: error.toString(),
stack: stack,
information: 'Context: $context',
);
}
Future<void> _persistCrash({
required String type,
required String error,
StackTrace? stack,
String? information,
}) async {
try {
final timestamp = DateTime.now().toIso8601String();
final crashData = StringBuffer()
..writeln('--- $timestamp ---')
..writeln('Type: $type')
..writeln('Error: $error')
..writeln('Stack: ${stack ?? 'no stack'}')
..writeln('Info: ${information ?? ''}')
..writeln(debugText())
..writeln()
..writeln();
final directory = await _getLogDirectory();
final file = File('${directory.path}/app.logs');
final fileLength = await file.length();
if (fileLength > 5 * 1024 * 1024) {
// If log file exceeds 5MB, truncate it
final lines = await file.readAsLines();
final half = lines.length ~/ 2;
final truncatedLines = lines.sublist(half);
await file.writeAsString(truncatedLines.join('\n'));
}
await file.writeAsString(crashData.toString(), mode: FileMode.append);
core.connection.signalNotification(LogNotification('App crashed: $error'));
} catch (_) {
// Avoid throwing from the crash logger
}
}
// Minimal implementation; customize per platform if needed.
Future<Directory> _getLogDirectory() async {
// On mobile, you might choose applicationDocumentsDirectory via platform channel,
// but staying pure Dart, use currentDirectory as a placeholder.
return Directory.current;
}
enum ConnectionType {
@@ -34,44 +140,83 @@ enum ConnectionType {
remote,
}
Future<void> initializeActions(ConnectionType connectionType) async {
void initializeActions(ConnectionType connectionType) {
if (kIsWeb) {
actionHandler = StubActions();
core.actionHandler = StubActions();
} else if (Platform.isAndroid) {
actionHandler = switch (connectionType) {
core.actionHandler = switch (connectionType) {
ConnectionType.local => AndroidActions(),
ConnectionType.remote => RemoteActions(),
ConnectionType.unknown => StubActions(),
};
} else if (Platform.isIOS) {
actionHandler = switch (connectionType) {
core.actionHandler = switch (connectionType) {
ConnectionType.local => StubActions(),
ConnectionType.remote => RemoteActions(),
ConnectionType.unknown => StubActions(),
};
} else {
actionHandler = switch (connectionType) {
core.actionHandler = switch (connectionType) {
ConnectionType.local => DesktopActions(),
ConnectionType.remote => RemoteActions(),
ConnectionType.unknown => StubActions(),
};
}
actionHandler.init(settings.getKeyMap());
core.actionHandler.init(core.settings.getKeyMap());
}
class SwiftPlayApp extends StatelessWidget {
const SwiftPlayApp({super.key});
class BikeControlApp extends StatelessWidget {
final Widget? customChild;
final BCPage page;
final String? error;
const BikeControlApp({super.key, this.error, this.page = BCPage.devices, this.customChild});
@override
Widget build(BuildContext context) {
return MaterialApp(
final isMobile = MediaQuery.sizeOf(context).width < 600;
return ShadcnApp(
navigatorKey: navigatorKey,
debugShowCheckedModeBanner: false,
title: 'SwiftControl',
theme: AppTheme.light,
darkTheme: AppTheme.dark,
themeMode: ThemeMode.light,
home: const RequirementsPage(),
menuHandler: PopoverOverlayHandler(),
popoverHandler: PopoverOverlayHandler(),
localizationsDelegates: [
...GlobalMaterialLocalizations.delegates,
GlobalWidgetsLocalizations.delegate,
ShadcnLocalizations.delegate,
AppLocalizations.delegate,
],
supportedLocales: AppLocalizations.delegate.supportedLocales,
title: 'BikeControl',
darkTheme: ThemeData(
colorScheme: ColorSchemes.darkDefaultColor.copyWith(
card: () => Color(0xFF001A29),
background: () => Color(0xFF232323),
muted: () => Color(0xFF3A3A3A),
),
),
theme: ThemeData(
colorScheme: ColorSchemes.lightDefaultColor.copyWith(
card: () => BKColor.background,
),
),
//themeMode: ThemeMode.dark,
home: error != null
? Center(
child: Text(
'There was an error starting the App. Please contact support:\n$error',
style: TextStyle(color: Colors.white),
),
)
: ToastLayer(
key: ValueKey('Test'),
padding: isMobile ? EdgeInsets.only(bottom: 60, left: 24, right: 24, top: 60) : null,
child: Stack(
children: [
customChild ?? Navigation(page: page),
Positioned.fill(child: Testbed()),
],
),
),
);
}
}

540
lib/pages/button_edit.dart Normal file
View File

@@ -0,0 +1,540 @@
import 'dart:async';
import 'package:bike_control/bluetooth/messages/notification.dart';
import 'package:bike_control/gen/l10n.dart';
import 'package:bike_control/pages/touch_area.dart';
import 'package:bike_control/utils/actions/android.dart';
import 'package:bike_control/utils/actions/base_actions.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/i18n_extension.dart';
import 'package:bike_control/utils/keymap/apps/custom_app.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:bike_control/utils/keymap/keymap.dart';
import 'package:bike_control/widgets/custom_keymap_selector.dart';
import 'package:bike_control/widgets/ui/button_widget.dart';
import 'package:bike_control/widgets/ui/colored_title.dart';
import 'package:bike_control/widgets/ui/colors.dart';
import 'package:bike_control/widgets/ui/warning.dart';
import 'package:dartx/dartx.dart';
import 'package:flutter/services.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
class ButtonEditPage extends StatefulWidget {
final Keymap keymap;
final KeyPair keyPair;
final VoidCallback onUpdate;
const ButtonEditPage({super.key, required this.keyPair, required this.onUpdate, required this.keymap});
@override
State<ButtonEditPage> createState() => _ButtonEditPageState();
}
class _ButtonEditPageState extends State<ButtonEditPage> {
late KeyPair _keyPair;
late final ScrollController _scrollController = ScrollController();
final double baseHeight = 46;
bool _bumped = false;
void _triggerBump() async {
setState(() {
_bumped = true;
});
await Future.delayed(const Duration(milliseconds: 150));
if (mounted) {
setState(() {
_bumped = false;
});
}
}
late StreamSubscription<BaseNotification> _actionSubscription;
@override
void initState() {
super.initState();
_keyPair = widget.keyPair;
_actionSubscription = core.connection.actionStream.listen((data) async {
if (!mounted) {
return;
}
if (data is ButtonNotification && data.buttonsClicked.length == 1) {
final clickedButton = data.buttonsClicked.first;
final keyPair = widget.keymap.keyPairs.firstOrNullWhere(
(kp) => kp.buttons.contains(clickedButton),
);
if (keyPair != null) {
setState(() {
_keyPair = keyPair;
});
_triggerBump();
}
}
});
}
@override
void dispose() {
_scrollController.dispose();
_actionSubscription.cancel();
super.dispose();
}
@override
Widget build(BuildContext context) {
final trainerApp = core.settings.getTrainerApp();
final actionsWithInGameAction = trainerApp?.keymap.keyPairs
.where((kp) => kp.inGameAction != null)
.distinctBy((kp) => kp.inGameAction)
.toList();
return IntrinsicWidth(
child: Scrollbar(
controller: _scrollController,
child: SingleChildScrollView(
controller: _scrollController,
child: Container(
constraints: BoxConstraints(maxWidth: 300),
padding: const EdgeInsets.only(right: 26.0),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
spacing: 8,
children: [
SizedBox(height: 16),
Row(
mainAxisSize: MainAxisSize.min,
mainAxisAlignment: MainAxisAlignment.center,
crossAxisAlignment: CrossAxisAlignment.center,
spacing: 8,
children: [
Text('Editing').h3,
AnimatedContainer(
duration: const Duration(milliseconds: 600),
curve: Curves.easeOut,
width: _keyPair.buttons.first.color != null ? baseHeight : null,
height: _keyPair.buttons.first.color != null ? baseHeight : null,
padding: EdgeInsets.all(_bumped ? 0 : 6.0),
constraints: BoxConstraints(maxWidth: 120),
child: ButtonWidget(button: _keyPair.buttons.first),
),
Expanded(child: SizedBox()),
IconButton(
icon: Icon(Icons.close),
variance: ButtonVariance.ghost,
onPressed: () {
closeDrawer(context);
},
),
],
),
if (core.logic.hasNoConnectionMethod)
ConstrainedBox(
constraints: BoxConstraints(maxWidth: 300),
child: Warning(
children: [
Text(AppLocalizations.of(context).pleaseSelectAConnectionMethodFirst),
],
),
),
if (core.logic.showObpActions) ...[
ColoredTitle(text: context.i18n.openBikeControlActions),
if (core.logic.obpConnectedApp == null)
Warning(
children: [
Text(
core.logic.obpConnectedApp == null
? 'Please connect to ${core.settings.getTrainerApp()?.name}, first.'
: context.i18n.appIdActions(core.logic.obpConnectedApp!.appId),
),
],
)
else
..._buildTrainerConnectionActions(core.logic.obpConnectedApp!.supportedActions),
],
if (core.settings.getMyWhooshLinkEnabled() && core.logic.showMyWhooshLink) ...[
SizedBox(height: 8),
ColoredTitle(text: context.i18n.myWhooshDirectConnectAction),
..._buildTrainerConnectionActions(core.whooshLink.supportedActions),
],
if (core.logic.isZwiftBleEnabled || core.logic.isZwiftMdnsEnabled) ...[
SizedBox(height: 8),
ColoredTitle(text: context.i18n.zwiftControllerAction),
..._buildTrainerConnectionActions(core.zwiftEmulator.supportedActions),
],
if (core.logic.showLocalRemoteOptions) ...[
SizedBox(height: 8),
ColoredTitle(text: 'Local / Remote Setting'),
if (trainerApp != null && trainerApp is! CustomApp && actionsWithInGameAction?.isEmpty != true) ...[
Builder(
builder: (context) => SelectableCard(
icon: null,
title: Text(context.i18n.predefinedAction(trainerApp.name)),
isActive: false,
onPressed: () {
showDropdown(
context: context,
builder: (c) => DropdownMenu(
children: actionsWithInGameAction!.map((keyPairAction) {
return MenuButton(
leading: keyPairAction.inGameAction?.icon != null
? Icon(keyPairAction.inGameAction!.icon)
: null,
onPressed: (_) {
// Copy all properties from the selected predefined action
if (core.actionHandler.supportedModes.contains(SupportedMode.keyboard)) {
_keyPair.physicalKey = keyPairAction.physicalKey;
_keyPair.logicalKey = keyPairAction.logicalKey;
_keyPair.modifiers = List.of(keyPairAction.modifiers);
} else {
_keyPair.physicalKey = null;
_keyPair.logicalKey = null;
_keyPair.modifiers = [];
}
if (core.actionHandler.supportedModes.contains(SupportedMode.touch)) {
_keyPair.touchPosition = keyPairAction.touchPosition;
} else {
_keyPair.touchPosition = Offset.zero;
}
_keyPair.isLongPress = keyPairAction.isLongPress;
_keyPair.inGameAction = keyPairAction.inGameAction;
_keyPair.inGameActionValue = keyPairAction.inGameActionValue;
setState(() {});
},
child: Text(keyPairAction.toString()),
);
}).toList(),
),
);
},
),
),
],
if (core.actionHandler.supportedModes.contains(SupportedMode.keyboard))
SelectableCard(
icon: RadixIcons.keyboard,
title: Text(context.i18n.simulateKeyboardShortcut),
isActive: _keyPair.physicalKey != null && !_keyPair.isSpecialKey,
value: _keyPair.toString(),
onPressed: () async {
await showDialog<void>(
context: context,
barrierDismissible: false, // enable Escape key
builder: (c) => HotKeyListenerDialog(
customApp: core.actionHandler.supportedApp! as CustomApp,
keyPair: _keyPair,
),
);
setState(() {});
widget.onUpdate();
},
),
if (core.actionHandler.supportedModes.contains(SupportedMode.touch))
SelectableCard(
title: Text(context.i18n.simulateTouch),
icon: core.actionHandler is AndroidActions ? Icons.touch_app_outlined : BootstrapIcons.mouse,
isActive: _keyPair.physicalKey == null && _keyPair.touchPosition != Offset.zero,
value: _keyPair.toString(),
onPressed: () async {
if (_keyPair.touchPosition == Offset.zero) {
_keyPair.touchPosition = Offset(50, 50);
}
_keyPair.physicalKey = null;
_keyPair.logicalKey = null;
await Navigator.of(context).push<bool?>(
MaterialPageRoute(
builder: (c) => TouchAreaSetupPage(
keyPair: _keyPair,
),
),
);
setState(() {});
widget.onUpdate();
},
),
if (core.actionHandler.supportedModes.contains(SupportedMode.media))
Builder(
builder: (context) => SelectableCard(
icon: Icons.music_note_outlined,
isActive: _keyPair.isSpecialKey,
title: Text(context.i18n.simulateMediaKey),
value: _keyPair.toString(),
onPressed: () {
showDropdown(
context: context,
builder: (c) => DropdownMenu(
children: [
MenuButton(
child: Text(context.i18n.playPause),
onPressed: (c) {
_keyPair.physicalKey = PhysicalKeyboardKey.mediaPlayPause;
_keyPair.logicalKey = null;
setState(() {});
widget.onUpdate();
},
),
MenuButton(
child: Text(context.i18n.stop),
onPressed: (c) {
_keyPair.physicalKey = PhysicalKeyboardKey.mediaStop;
_keyPair.logicalKey = null;
setState(() {});
widget.onUpdate();
},
),
MenuButton(
child: Text(context.i18n.previous),
onPressed: (c) {
_keyPair.physicalKey = PhysicalKeyboardKey.mediaTrackPrevious;
_keyPair.logicalKey = null;
setState(() {});
widget.onUpdate();
},
),
MenuButton(
child: Text(context.i18n.next),
onPressed: (c) {
_keyPair.physicalKey = PhysicalKeyboardKey.mediaTrackNext;
_keyPair.logicalKey = null;
setState(() {});
widget.onUpdate();
},
),
MenuButton(
onPressed: (c) {
_keyPair.physicalKey = PhysicalKeyboardKey.audioVolumeUp;
_keyPair.logicalKey = null;
setState(() {});
widget.onUpdate();
},
child: Text(context.i18n.volumeUp),
),
MenuButton(
child: Text(context.i18n.volumeDown),
onPressed: (c) {
_keyPair.physicalKey = PhysicalKeyboardKey.audioVolumeDown;
_keyPair.logicalKey = null;
setState(() {});
widget.onUpdate();
},
),
],
),
);
},
),
),
],
if (core.connection.accessories.isNotEmpty) ...[
SizedBox(height: 8),
ColoredTitle(text: 'Accessory Actions'),
Builder(
builder: (context) => SelectableCard(
icon: Icons.air,
title: Text('KICKR Headwind'),
isActive:
_keyPair.inGameAction != null &&
(_keyPair.inGameAction == InGameAction.headwindSpeed ||
_keyPair.inGameAction == InGameAction.headwindHeartRateMode),
value: _keyPair.inGameAction != null
? '${_keyPair.inGameAction} ${_keyPair.inGameActionValue ?? ""}'.trim()
: null,
onPressed: () {
showDropdown(
context: context,
builder: (c) => DropdownMenu(
children: [
MenuButton(
subMenu: [0, 25, 50, 75, 100]
.map(
(value) => MenuButton(
child: Text('Set Speed to $value%'),
onPressed: (_) {
_keyPair.inGameAction = InGameAction.headwindSpeed;
_keyPair.inGameActionValue = value;
widget.onUpdate();
setState(() {});
},
),
)
.toList(),
child: Text('Set Speed'),
),
MenuButton(
child: Text('Set to Heart Rate Mode'),
onPressed: (_) {
_keyPair.inGameAction = InGameAction.headwindHeartRateMode;
_keyPair.inGameActionValue = null;
widget.onUpdate();
setState(() {});
},
),
],
),
);
},
),
),
],
SizedBox(height: 8),
ColoredTitle(text: context.i18n.setting),
SelectableCard(
icon: _keyPair.isLongPress ? Icons.check_box : Icons.check_box_outline_blank,
title: Text(context.i18n.longPressMode),
isActive: _keyPair.isLongPress,
onPressed: () {
_keyPair.isLongPress = !_keyPair.isLongPress;
widget.onUpdate();
setState(() {});
},
),
SizedBox(height: 8),
DestructiveButton(
onPressed: () {
_keyPair.isLongPress = false;
_keyPair.physicalKey = null;
_keyPair.logicalKey = null;
_keyPair.modifiers = [];
_keyPair.touchPosition = Offset.zero;
_keyPair.inGameAction = null;
_keyPair.inGameActionValue = null;
widget.onUpdate();
setState(() {});
},
child: Text(context.i18n.unassignAction),
),
SizedBox(height: 16),
],
),
),
),
),
);
}
List<Widget> _buildTrainerConnectionActions(List<InGameAction> supportedActions) {
return supportedActions.map((action) {
return Builder(
builder: (context) {
return SelectableCard(
icon: action.icon,
title: Text(action.title),
subtitle: (action.possibleValues != null && action == _keyPair.inGameAction)
? Text(_keyPair.inGameActionValue!.toString())
: null,
isActive: _keyPair.inGameAction == action && supportedActions.contains(_keyPair.inGameAction),
onPressed: () {
if (action.possibleValues?.isNotEmpty == true) {
showDropdown(
context: context,
builder: (c) => DropdownMenu(
children: action.possibleValues!.map(
(ingame) {
return MenuButton(
child: Text(ingame.toString()),
onPressed: (_) {
_keyPair.touchPosition = Offset.zero;
_keyPair.physicalKey = null;
_keyPair.logicalKey = null;
_keyPair.inGameAction = action;
_keyPair.inGameActionValue = ingame;
widget.onUpdate();
setState(() {});
},
);
},
).toList(),
),
);
} else {
_keyPair.touchPosition = Offset.zero;
_keyPair.physicalKey = null;
_keyPair.logicalKey = null;
_keyPair.inGameAction = action;
_keyPair.inGameActionValue = null;
widget.onUpdate();
setState(() {});
}
},
);
},
);
}).toList();
}
}
class SelectableCard extends StatelessWidget {
final Widget title;
final Widget? subtitle;
final IconData? icon;
final bool isActive;
final String? value;
final VoidCallback? onPressed;
const SelectableCard({
super.key,
required this.title,
this.icon,
this.subtitle,
required this.isActive,
this.value,
required this.onPressed,
});
@override
Widget build(BuildContext context) {
return Button.outline(
style:
ButtonStyle(
variance: ButtonVariance.outline,
)
.withBorder(
border: isActive
? Border.all(color: BKColor.main, width: 2)
: Border.all(color: Theme.of(context).colorScheme.border, width: 2),
hoverBorder: Border.all(color: BKColor.mainEnd, width: 2),
focusBorder: Border.all(color: BKColor.main, width: 2),
)
.withBackgroundColor(
color: isActive
? Theme.of(context).brightness == Brightness.dark
? Theme.of(context).colorScheme.card
: Theme.of(context).colorScheme.card.withLuminance(0.9)
: Theme.of(context).colorScheme.background,
hoverColor: Theme.of(context).colorScheme.card,
),
onPressed: onPressed,
alignment: Alignment.centerLeft,
child: Container(
width: double.infinity,
padding: const EdgeInsets.symmetric(vertical: 8.0),
child: Basic(
leading: icon != null
? Padding(
padding: const EdgeInsets.only(top: 3.0),
child: Icon(
icon,
color: icon == Icons.delete_outline ? Theme.of(context).colorScheme.destructive : null,
),
)
: null,
title: title,
subtitle: value != null && isActive ? Text(value!) : subtitle,
),
),
);
}
}

View File

@@ -0,0 +1,690 @@
import 'dart:math';
import 'package:bike_control/bluetooth/devices/mywhoosh/link.dart';
import 'package:bike_control/bluetooth/devices/openbikecontrol/obc_ble_emulator.dart';
import 'package:bike_control/bluetooth/devices/openbikecontrol/obc_mdns_emulator.dart';
import 'package:bike_control/bluetooth/devices/trainer_connection.dart';
import 'package:bike_control/bluetooth/devices/zwift/ftms_mdns_emulator.dart';
import 'package:bike_control/bluetooth/devices/zwift/zwift_emulator.dart';
import 'package:bike_control/bluetooth/remote_pairing.dart';
import 'package:bike_control/main.dart';
import 'package:bike_control/pages/touch_area.dart';
import 'package:bike_control/utils/actions/android.dart';
import 'package:bike_control/utils/actions/base_actions.dart';
import 'package:bike_control/utils/actions/desktop.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/i18n_extension.dart';
import 'package:bike_control/utils/iap/iap_manager.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:bike_control/utils/keymap/keymap.dart';
import 'package:bike_control/widgets/apps/mywhoosh_link_tile.dart';
import 'package:bike_control/widgets/apps/openbikecontrol_ble_tile.dart';
import 'package:bike_control/widgets/apps/openbikecontrol_mdns_tile.dart';
import 'package:bike_control/widgets/apps/zwift_mdns_tile.dart';
import 'package:bike_control/widgets/apps/zwift_tile.dart';
import 'package:bike_control/widgets/pair_widget.dart';
import 'package:bike_control/widgets/ui/gradient_text.dart';
import 'package:bike_control/widgets/ui/toast.dart';
import 'package:bike_control/widgets/ui/warning.dart';
import 'package:dartx/dartx.dart';
import 'package:flutter/material.dart' show BackButton;
import 'package:flutter/services.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
class ButtonSimulator extends StatefulWidget {
const ButtonSimulator({super.key});
@override
State<ButtonSimulator> createState() => _ButtonSimulatorState();
}
class _ButtonSimulatorState extends State<ButtonSimulator> {
late final FocusNode _focusNode;
Map<InGameAction, String> _hotkeys = {};
// Default hotkeys for actions
static const List<String> _defaultHotkeyOrder = [
'1',
'2',
'3',
'4',
'5',
'6',
'7',
'8',
'9',
'q',
'w',
'e',
'r',
't',
'y',
'u',
'i',
'o',
'p',
'a',
's',
'd',
'f',
'g',
'h',
'j',
'k',
'l',
'z',
'x',
'c',
'v',
'b',
'n',
'm',
];
static const Duration _keyPressDuration = Duration(milliseconds: 200);
InGameAction? _pressedAction;
DateTime? _lastDown;
@override
void initState() {
super.initState();
_focusNode = FocusNode(debugLabel: 'ButtonSimulatorFocus', canRequestFocus: true);
_loadHotkeys();
_focusNode.requestFocus();
}
@override
void dispose() {
_focusNode.dispose();
super.dispose();
}
Future<void> _loadHotkeys() async {
final savedHotkeys = core.settings.getButtonSimulatorHotkeys();
// If no saved hotkeys, initialize with defaults
if (savedHotkeys.isEmpty) {
final connectedTrainers = core.logic.enabledTrainerConnections;
final allActions = <InGameAction>[];
for (final connection in connectedTrainers) {
allActions.addAll(connection.supportedActions);
}
// Assign default hotkeys to actions
final Map<InGameAction, String> defaultHotkeys = {};
int hotkeyIndex = 0;
for (final action in allActions.distinct()) {
if (hotkeyIndex < _defaultHotkeyOrder.length) {
defaultHotkeys[action] = _defaultHotkeyOrder[hotkeyIndex];
hotkeyIndex++;
}
}
await core.settings.setButtonSimulatorHotkeys(defaultHotkeys);
if (mounted) {
setState(() {
_hotkeys = defaultHotkeys;
});
}
} else {
setState(() {
_hotkeys = savedHotkeys;
});
}
}
KeyEventResult _onKey(FocusNode node, KeyEvent event) {
if (event is! KeyDownEvent) return KeyEventResult.ignored;
final key = event.logicalKey.keyLabel.toLowerCase();
// Find the action associated with this key
final action = _hotkeys.entries.firstOrNullWhere((entry) => entry.value == key)?.key;
if (action == null) return KeyEventResult.ignored;
_pressedAction = action;
setState(() {});
// Find the connection that supports this action
final connectedTrainers = core.logic.connectedTrainerConnections;
final connection = connectedTrainers.firstOrNullWhere((c) => c.supportedActions.contains(action));
if (connection != null) {
_sendKey(context, down: true, action: action, connection: connection);
// Schedule key up event
Future.delayed(
_keyPressDuration,
() {
if (mounted) {
_pressedAction = null;
setState(() {});
_sendKey(context, down: false, action: action, connection: connection);
}
},
);
return KeyEventResult.handled;
} else {
_pressedAction = null;
setState(() {});
buildToast(context, title: 'No connected trainer.');
}
return KeyEventResult.ignored;
}
@override
Widget build(BuildContext context) {
final connectedTrainers = core.logic.enabledTrainerConnections;
final isMobile = MediaQuery.sizeOf(context).width < 600;
return Focus(
focusNode: _focusNode,
autofocus: true,
onKeyEvent: _onKey,
child: Scaffold(
headers: [
AppBar(
leading: [BackButton()],
title: Text(context.i18n.simulateButtons),
trailing: [
PrimaryButton(
child: Icon(Icons.settings),
onPressed: () => _showHotkeySettings(context, connectedTrainers),
),
],
),
],
child: Scrollbar(
child: SingleChildScrollView(
padding: EdgeInsets.all(16),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
spacing: 16,
children: [
if (connectedTrainers.isEmpty)
Warning(
children: [
Text('No suitable connection method activated. Connect a trainer to simulate button presses.'),
],
),
for (final connectedTrainer in connectedTrainers)
if (!screenshotMode)
switch (connectedTrainer.title) {
WhooshLink.connectionTitle => MyWhooshLinkTile(),
ZwiftEmulator.connectionTitle => ZwiftTile(
onUpdate: () {
if (mounted) setState(() {});
},
),
FtmsMdnsEmulator.connectionTitle => ZwiftMdnsTile(
onUpdate: () {
setState(() {});
},
),
OpenBikeControlMdnsEmulator.connectionTitle => OpenBikeControlMdnsTile(),
OpenBikeControlBluetoothEmulator.connectionTitle => OpenBikeControlBluetoothTile(),
RemotePairing.connectionTitle => RemotePairingWidget(),
_ => SizedBox.shrink(),
},
...connectedTrainers.map(
(connection) {
final supportedActions = connection.supportedActions;
final actionGroups = {
if (supportedActions.contains(InGameAction.shiftUp) &&
supportedActions.contains(InGameAction.shiftDown))
'Shifting': [InGameAction.shiftDown, InGameAction.shiftUp],
'Other': supportedActions
.where(
(action) =>
action != InGameAction.shiftUp &&
action != InGameAction.shiftDown &&
action != InGameAction.steerLeft &&
action != InGameAction.steerRight,
)
.toList(),
if (supportedActions.contains(InGameAction.steerLeft) &&
supportedActions.contains(InGameAction.steerRight))
'Steering': [InGameAction.steerLeft, InGameAction.steerRight],
};
return [
GradientText(connection.title).bold.large,
ConstrainedBox(
constraints: BoxConstraints(maxWidth: 800),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
spacing: 12,
children: [
for (final group in actionGroups.entries) ...[
Text(group.key.toUpperCase()).bold.muted,
if (group.value.length == 2)
Row(
spacing: 8,
children: group.value.map(
(action) {
final hotkey = _hotkeys[action];
return Expanded(
child: Stack(
children: [
SizedBox(
height: 150,
width: double.infinity,
child: _buildButton(action, group, connection, isMobile),
),
if (hotkey != null)
Positioned(
top: -4,
right: -4,
child: KeyWidget(
label: hotkey.toUpperCase(),
invert: true,
),
),
],
),
);
},
).toList(),
)
else
GridView.count(
shrinkWrap: true,
physics: NeverScrollableScrollPhysics(),
crossAxisSpacing: 8,
mainAxisSpacing: 8,
crossAxisCount: min(group.value.length, 3),
childAspectRatio: isMobile ? 1 : 2.4,
children: group.value.map(
(action) {
final hotkey = _hotkeys[action];
return Stack(
fit: StackFit.expand,
children: [
_buildButton(action, group, connection, isMobile),
if (hotkey != null)
Positioned(
top: -4,
right: -4,
child: KeyWidget(
label: hotkey.toUpperCase(),
),
),
],
);
},
).toList(),
),
SizedBox(height: 12),
],
],
),
),
];
},
).flatten(),
// local control doesn't make much sense - it would send the key events to BikeControl itself
if (false &&
core.logic.showLocalControl &&
core.settings.getLocalEnabled() &&
core.actionHandler.supportedApp != null) ...[
GradientText('Local Control'),
Wrap(
spacing: 12,
runSpacing: 12,
children: core.actionHandler.supportedApp!.keymap.keyPairs
.map(
(keyPair) => PrimaryButton(
child: Text(keyPair.toString()),
onPressed: () async {
if (core.actionHandler is AndroidActions) {
await (core.actionHandler as AndroidActions).performAction(
keyPair.buttons.first,
isKeyDown: true,
isKeyUp: false,
);
await (core.actionHandler as AndroidActions).performAction(
keyPair.buttons.first,
isKeyDown: false,
isKeyUp: true,
);
} else {
await (core.actionHandler as DesktopActions).performAction(
keyPair.buttons.first,
isKeyDown: true,
isKeyUp: false,
);
await (core.actionHandler as DesktopActions).performAction(
keyPair.buttons.first,
isKeyDown: false,
isKeyUp: true,
);
}
},
),
)
.toList(),
),
],
],
),
),
),
),
);
}
Widget _buildButton(
InGameAction action,
MapEntry<String, List<InGameAction>> group,
TrainerConnection connection,
bool isMobile,
) {
return Builder(
builder: (context) {
return Button(
style: _pressedAction == action
? ButtonStyle.outline()
: group.key == 'Other'
? ButtonStyle.outline()
: ButtonStyle.primary(),
child: Column(
crossAxisAlignment: CrossAxisAlignment.center,
mainAxisAlignment: MainAxisAlignment.center,
children: [
if (action.icon != null) ...[
Icon(action.icon),
SizedBox(height: 8),
],
Text(
action.title,
textAlign: TextAlign.center,
style: TextStyle(height: 1),
maxLines: 2,
).bold,
if (action.alternativeTitle != null)
Text(
action.alternativeTitle!.toUpperCase(),
style: TextStyle(fontSize: 10, color: Colors.gray),
),
],
),
onPressed: () {},
onTapDown: (c) async {
_sendKey(context, down: true, action: action, connection: connection);
/*final device = HidDevice('Simulator');
final button = ControllerButton('action', action: InGameAction.openActionBar);
device.getOrAddButton(button.name, () => button);
device.handleButtonsClickedWithoutLongPressSupport([button]);*/
},
onTapUp: (c) async {
_sendKey(context, down: false, action: action, connection: connection);
},
);
},
);
}
Future<void> _sendKey(
BuildContext context, {
required bool down,
required InGameAction action,
required TrainerConnection connection,
}) async {
if (!connection.isConnected.value) {
if (down) {
buildToast(context, title: 'No connected trainer.');
}
return;
}
if (action.possibleValues != null) {
if (down) return;
showDropdown(
context: context,
builder: (context) => DropdownMenu(
children: action.possibleValues!
.map(
(e) => MenuButton(
child: Text(e.toString()),
onPressed: (c) async {
await connection.sendAction(
KeyPair(
buttons: [],
physicalKey: null,
logicalKey: null,
inGameAction: action,
inGameActionValue: e,
),
isKeyDown: false,
isKeyUp: true,
);
},
),
)
.toList(),
),
);
return;
} else {
if (!down && _lastDown != null) {
final timeSinceLastDown = DateTime.now().difference(_lastDown!);
if (timeSinceLastDown < Duration(milliseconds: 400)) {
// wait a bit so actions actually get applied correctly for some trainer apps
await Future.delayed(Duration(milliseconds: 800) - timeSinceLastDown);
}
} else if (down) {
_lastDown = DateTime.now();
}
final result = await connection.sendAction(
KeyPair(
buttons: [],
physicalKey: null,
logicalKey: null,
inGameAction: action,
),
isKeyDown: down,
isKeyUp: !down,
);
await IAPManager.instance.incrementCommandCount();
if (result is! Success) {
buildToast(context, title: result.message);
}
}
}
void _showHotkeySettings(BuildContext context, List<TrainerConnection> connections) {
showDialog(
context: context,
builder: (context) => _HotkeySettingsDialog(
connections: connections,
currentHotkeys: _hotkeys,
onSave: (newHotkeys) {
setState(() {
_hotkeys = newHotkeys;
});
},
),
);
}
}
class _HotkeySettingsDialog extends StatefulWidget {
final List<TrainerConnection> connections;
final Map<InGameAction, String> currentHotkeys;
final Function(Map<InGameAction, String>) onSave;
const _HotkeySettingsDialog({
required this.connections,
required this.currentHotkeys,
required this.onSave,
});
@override
State<_HotkeySettingsDialog> createState() => _HotkeySettingsDialogState();
}
class _HotkeySettingsDialogState extends State<_HotkeySettingsDialog> {
late Map<InGameAction, String> _editableHotkeys;
InGameAction? _editingAction;
late FocusNode _focusNode;
static final _validHotkeyPattern = RegExp(r'[0-9a-z]');
@override
void initState() {
super.initState();
_editableHotkeys = Map.from(widget.currentHotkeys);
_focusNode = FocusNode(debugLabel: 'HotkeySettingsFocus', canRequestFocus: true);
}
@override
void dispose() {
_focusNode.dispose();
super.dispose();
}
KeyEventResult _onKey(FocusNode node, KeyEvent event) {
if (_editingAction == null || event is! KeyDownEvent) return KeyEventResult.ignored;
final key = event.logicalKey.keyLabel.toLowerCase();
// Only allow single character 1-9 and a-z
if (key.length == 1 && _validHotkeyPattern.hasMatch(key)) {
setState(() {
_editableHotkeys[_editingAction!] = key;
_editingAction = null;
});
return KeyEventResult.handled;
}
// Escape to cancel
if (event.logicalKey == LogicalKeyboardKey.escape) {
setState(() {
_editingAction = null;
});
return KeyEventResult.handled;
}
return KeyEventResult.ignored;
}
@override
Widget build(BuildContext context) {
final allActions = <InGameAction>[];
for (final connection in widget.connections) {
allActions.addAll(connection.supportedActions);
}
final uniqueActions = allActions.distinct().toList();
return Focus(
focusNode: _focusNode,
autofocus: true,
onKeyEvent: _onKey,
child: AlertDialog(
title: Text('Configure Keyboard Hotkeys'),
content: SizedBox(
width: 500,
child: Column(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.start,
spacing: 8,
children: [
Text('Assign keyboard shortcuts to simulator buttons').muted,
SizedBox(height: 8),
Flexible(
child: SingleChildScrollView(
child: Column(
spacing: 8,
children: uniqueActions.map((action) {
final hotkey = _editableHotkeys[action];
final isEditing = _editingAction == action;
return Card(
child: Container(
padding: EdgeInsets.all(12),
child: Row(
children: [
Expanded(
child: Text(action.title),
),
if (isEditing)
Container(
padding: EdgeInsets.symmetric(horizontal: 8, vertical: 4),
decoration: BoxDecoration(
border: Border.all(color: Colors.blue),
borderRadius: BorderRadius.circular(4),
),
child: Text('Press a key...', style: TextStyle(color: Colors.blue)),
)
else if (hotkey != null)
Container(
padding: EdgeInsets.symmetric(horizontal: 8, vertical: 4),
decoration: BoxDecoration(
color: Colors.gray.withOpacity(0.3),
borderRadius: BorderRadius.circular(4),
),
child: Text(hotkey.toUpperCase(), style: TextStyle(fontWeight: FontWeight.bold)),
)
else
Text('No hotkey', style: TextStyle(color: Colors.gray)),
SizedBox(width: 8),
OutlineButton(
size: ButtonSize.small,
child: Text(isEditing ? 'Cancel' : 'Set'),
onPressed: () {
setState(() {
_editingAction = isEditing ? null : action;
});
},
),
if (hotkey != null && !isEditing) ...[
SizedBox(width: 4),
OutlineButton(
size: ButtonSize.small,
child: Text('Clear'),
onPressed: () {
setState(() {
_editableHotkeys.remove(action);
});
},
),
],
],
),
),
);
}).toList(),
),
),
),
],
),
),
actions: [
SecondaryButton(
child: Text('Cancel'),
onPressed: () => Navigator.of(context).pop(),
),
PrimaryButton(
child: Text('Save'),
onPressed: () async {
await core.settings.setButtonSimulatorHotkeys(_editableHotkeys);
widget.onSave(_editableHotkeys);
if (context.mounted) {
Navigator.of(context).pop();
}
},
),
],
),
);
}
}

View File

@@ -0,0 +1,206 @@
import 'dart:async';
import 'dart:io';
import 'package:bike_control/gen/l10n.dart';
import 'package:bike_control/main.dart';
import 'package:bike_control/pages/button_edit.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/i18n_extension.dart';
import 'package:bike_control/utils/keymap/apps/custom_app.dart';
import 'package:bike_control/utils/keymap/apps/my_whoosh.dart';
import 'package:bike_control/utils/keymap/apps/supported_app.dart';
import 'package:bike_control/utils/requirements/multi.dart';
import 'package:bike_control/widgets/ui/colored_title.dart';
import 'package:bike_control/widgets/ui/warning.dart';
import 'package:flutter/foundation.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
class ConfigurationPage extends StatefulWidget {
final VoidCallback onUpdate;
const ConfigurationPage({super.key, required this.onUpdate});
@override
State<ConfigurationPage> createState() => _ConfigurationPageState();
}
class _ConfigurationPageState extends State<ConfigurationPage> {
@override
Widget build(BuildContext context) {
return Column(
spacing: 12,
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
ColoredTitle(text: context.i18n.setupTrainer),
Card(
fillColor: Theme.of(context).colorScheme.background,
filled: true,
borderWidth: 1,
borderColor: Theme.of(context).colorScheme.border,
child: Builder(
builder: (context) {
return StatefulBuilder(
builder: (c, setState) => Column(
spacing: 8,
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Select<SupportedApp>(
constraints: BoxConstraints(maxWidth: 400, minWidth: 400),
itemBuilder: (c, app) => Row(
spacing: 4,
children: [
Text(screenshotMode ? 'Trainer app' : app.name),
if (app.supportsOpenBikeProtocol) Icon(Icons.star),
],
),
popup: SelectPopup(
items: SelectItemList(
children: SupportedApp.supportedApps.map((app) {
return SelectItemButton(
value: app,
child: Row(
spacing: 4,
children: [
Text(app.name),
if (app.supportsOpenBikeProtocol) Icon(Icons.star),
],
),
);
}).toList(),
),
).call,
placeholder: Text(context.i18n.selectTrainerAppPlaceholder),
value: core.settings.getTrainerApp(),
onChanged: (selectedApp) async {
if (selectedApp is! MyWhoosh) {
if (core.whooshLink.isStarted.value) {
core.whooshLink.stopServer();
}
}
if (!selectedApp!.supportsZwiftEmulation) {
if (core.zwiftMdnsEmulator.isStarted.value) {
core.zwiftMdnsEmulator.stop();
}
if (core.zwiftEmulator.isStarted.value) {
core.zwiftEmulator.stopAdvertising();
}
}
if (!selectedApp.supportsOpenBikeProtocol) {
if (core.obpMdnsEmulator.isStarted.value) {
core.obpMdnsEmulator.stopServer();
}
if (core.obpBluetoothEmulator.isStarted.value) {
core.obpBluetoothEmulator.stopServer();
}
}
core.settings.setTrainerApp(selectedApp);
if (core.settings.getLastTarget() == null && Target.thisDevice.isCompatible) {
await _setTarget(context, Target.thisDevice);
} else if (core.settings.getLastTarget() == null && Target.otherDevice.isCompatible) {
await _setTarget(context, Target.otherDevice);
}
if (core.actionHandler.supportedApp == null ||
(core.actionHandler.supportedApp is! CustomApp && selectedApp is! CustomApp)) {
core.actionHandler.init(selectedApp);
core.settings.setKeyMap(selectedApp);
}
widget.onUpdate();
setState(() {});
},
),
if (core.settings.getTrainerApp() != null) ...[
if (core.settings.getTrainerApp()!.supportsOpenBikeProtocol == true && !screenshotMode)
Text(
AppLocalizations.of(context).openBikeControlAnnouncement(core.settings.getTrainerApp()!.name),
).xSmall,
SizedBox(height: 0),
Text(
context.i18n.selectTargetWhereAppRuns(
screenshotMode ? 'Trainer app' : core.settings.getTrainerApp()?.name ?? 'the Trainer app',
),
).small,
Row(
spacing: 8,
children: [Target.thisDevice, Target.otherDevice]
.map(
(target) => Expanded(
child: SelectableCard(
title: Center(child: Icon(target.icon)),
isActive: target == core.settings.getLastTarget(),
subtitle: Center(
child: Column(
children: [
Text(target.getTitle(context)),
if (!target.isCompatible) Text(context.i18n.platformRestrictionNotSupported),
],
),
),
onPressed: !target.isCompatible
? null
: () async {
await _setTarget(context, target);
setState(() {});
widget.onUpdate();
},
),
),
)
.toList(),
),
],
if (core.settings.getLastTarget() == Target.otherDevice &&
!core.logic.hasRecommendedConnectionMethods) ...[
SizedBox(height: 8),
Warning(
children: [
Text(
'BikeControl is available on iOS, Android, Windows and macOS. For proper support for ${core.settings.getTrainerApp()?.name} please download BikeControl on that device.',
).small,
],
),
],
if (core.settings.getTrainerApp()?.star == true && !screenshotMode)
Row(
spacing: 8,
children: [
Icon(Icons.star),
Expanded(
child: Text(
AppLocalizations.of(
context,
).newConnectionMethodAnnouncement(core.settings.getTrainerApp()!.name),
style: TextStyle(fontWeight: FontWeight.bold),
).xSmall,
),
],
),
],
),
);
},
),
),
],
);
}
Future<void> _setTarget(BuildContext context, Target target) async {
await core.settings.setLastTarget(target);
if (core.settings.getTrainerApp()?.supportsOpenBikeProtocol == true && !core.logic.emulatorEnabled) {
core.settings.setObpMdnsEnabled(true);
}
// enable local connection on Windows if the app doesn't support OBP
if (target == Target.thisDevice &&
core.settings.getTrainerApp()?.supportsOpenBikeProtocol == false &&
!kIsWeb &&
Platform.isWindows) {
core.settings.setLocalEnabled(true);
}
core.logic.startEnabledConnectionMethod();
}
}

165
lib/pages/customize.dart Normal file
View File

@@ -0,0 +1,165 @@
import 'package:bike_control/main.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/i18n_extension.dart';
import 'package:bike_control/utils/iap/iap_manager.dart';
import 'package:bike_control/utils/keymap/apps/custom_app.dart';
import 'package:bike_control/utils/keymap/apps/supported_app.dart';
import 'package:bike_control/utils/keymap/manager.dart';
import 'package:bike_control/widgets/iap_status_widget.dart';
import 'package:bike_control/widgets/keymap_explanation.dart';
import 'package:bike_control/widgets/ui/beta_pill.dart';
import 'package:bike_control/widgets/ui/colored_title.dart';
import 'package:bike_control/widgets/ui/warning.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
class CustomizePage extends StatefulWidget {
const CustomizePage({super.key});
@override
State<CustomizePage> createState() => _CustomizeState();
}
class _CustomizeState extends State<CustomizePage> {
@override
Widget build(BuildContext context) {
return SingleChildScrollView(
padding: EdgeInsets.all(16),
child: Column(
spacing: 12,
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
ValueListenableBuilder(
valueListenable: IAPManager.instance.isPurchased,
builder: (context, value, child) => value ? SizedBox.shrink() : IAPStatusWidget(small: true),
),
Container(
margin: const EdgeInsets.only(bottom: 8.0),
padding: const EdgeInsets.symmetric(vertical: 8.0),
width: double.infinity,
child: ColoredTitle(
text: context.i18n.customizeControllerButtons(
screenshotMode ? 'Trainer app' : (core.settings.getTrainerApp()?.name ?? ''),
),
),
),
Select<SupportedApp?>(
constraints: BoxConstraints(minWidth: 300),
value: core.actionHandler.supportedApp,
popup: SelectPopup(
items: SelectItemList(
children: [
..._getAllApps().map(
(a) => SelectItemButton(
value: a,
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Expanded(child: Text(a.name)),
if (a is CustomApp)
BetaPill(text: 'CUSTOM')
else if (a.supportsOpenBikeProtocol)
Icon(Icons.star, size: 16),
],
),
),
),
SelectItemButton(
value: CustomApp(profileName: 'New'),
child: Row(
spacing: 6,
children: [
Icon(Icons.add, color: Theme.of(context).colorScheme.mutedForeground),
Expanded(child: Text(context.i18n.createNewKeymap).normal.muted),
],
),
),
],
),
).call,
itemBuilder: (c, app) => Row(
spacing: 8,
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Expanded(child: Text(screenshotMode ? 'Trainer app' : app!.name)),
if (app is CustomApp) BetaPill(text: 'CUSTOM'),
],
),
placeholder: Text(context.i18n.selectKeymap),
onChanged: (app) async {
if (app == null) {
return;
} else if (app.name == 'New') {
final profileName = await KeymapManager().showNewProfileDialog(context);
if (profileName != null && profileName.isNotEmpty) {
final customApp = CustomApp(profileName: profileName);
core.actionHandler.init(customApp);
await core.settings.setKeyMap(customApp);
setState(() {});
}
} else {
core.actionHandler.init(app);
await core.settings.setKeyMap(app);
setState(() {});
}
},
),
KeymapManager().getManageProfileDialog(
context,
core.actionHandler.supportedApp is CustomApp ? core.actionHandler.supportedApp?.name : null,
onDone: () {
setState(() {});
},
),
if (core.actionHandler.supportedApp is! CustomApp)
Text(
context.i18n.customizeKeymapHint,
style: TextStyle(fontSize: 12),
),
Gap(12),
if (core.actionHandler.supportedApp != null && core.connection.controllerDevices.isNotEmpty)
KeymapExplanation(
key: Key(core.actionHandler.supportedApp!.keymap.runtimeType.toString()),
keymap: core.actionHandler.supportedApp!.keymap,
onUpdate: () {
setState(() {});
if (core.actionHandler.supportedApp is CustomApp) {
core.settings.setKeyMap(core.actionHandler.supportedApp!);
}
},
)
else if (core.connection.controllerDevices.isEmpty)
Warning(
children: [Text(context.i18n.connectControllerToPreview).small],
),
],
),
);
}
List<SupportedApp> _getAllApps() {
final baseApp = core.settings.getTrainerApp();
final customProfiles = core.settings.getCustomAppProfiles();
final customApps = customProfiles.map((profile) {
final customApp = CustomApp(profileName: profile);
final savedKeymap = core.settings.getCustomAppKeymap(profile);
if (savedKeymap != null) {
customApp.decodeKeymap(savedKeymap);
}
return customApp;
}).toList();
// If no custom profiles exist, add the default "Custom" one
if (customApps.isEmpty) {
customApps.add(CustomApp());
}
return [if (baseApp != null) baseApp, ...customApps];
}
}

View File

@@ -1,42 +1,20 @@
import 'dart:async';
import 'dart:io';
import 'package:device_auto_rotate_checker/device_auto_rotate_checker.dart';
import 'package:device_info_plus/device_info_plus.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:package_info_plus/package_info_plus.dart';
import 'package:swift_control/bluetooth/devices/link/link_device.dart';
import 'package:swift_control/bluetooth/devices/zwift/zwift_emulator.dart';
import 'package:swift_control/main.dart';
import 'package:swift_control/utils/actions/desktop.dart';
import 'package:swift_control/utils/keymap/apps/my_whoosh.dart';
import 'package:swift_control/utils/keymap/manager.dart';
import 'package:swift_control/utils/requirements/multi.dart';
import 'package:swift_control/utils/requirements/zwift.dart';
import 'package:swift_control/widgets/beta_pill.dart';
import 'package:swift_control/widgets/keymap_explanation.dart';
import 'package:swift_control/widgets/logviewer.dart';
import 'package:swift_control/widgets/scan.dart';
import 'package:swift_control/widgets/small_progress_indicator.dart';
import 'package:swift_control/widgets/testbed.dart';
import 'package:swift_control/widgets/title.dart';
import 'package:swift_control/widgets/warning.dart';
import 'package:universal_ble/universal_ble.dart';
import 'package:url_launcher/url_launcher.dart';
import 'package:wakelock_plus/wakelock_plus.dart';
import 'package:bike_control/gen/l10n.dart';
import 'package:bike_control/main.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/i18n_extension.dart';
import 'package:bike_control/utils/iap/iap_manager.dart';
import 'package:bike_control/widgets/iap_status_widget.dart';
import 'package:bike_control/widgets/scan.dart';
import 'package:bike_control/widgets/ui/colored_title.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
import '../bluetooth/devices/base_device.dart';
import '../utils/actions/android.dart';
import '../utils/actions/remote.dart';
import '../utils/keymap/apps/custom_app.dart';
import '../utils/keymap/apps/supported_app.dart';
import '../utils/requirements/remote.dart';
import '../widgets/changelog_dialog.dart';
import '../widgets/menu.dart';
class DevicePage extends StatefulWidget {
const DevicePage({super.key});
final VoidCallback onUpdate;
const DevicePage({super.key, required this.onUpdate});
@override
State<DevicePage> createState() => _DevicePageState();
@@ -44,515 +22,97 @@ class DevicePage extends StatefulWidget {
class _DevicePageState extends State<DevicePage> with WidgetsBindingObserver {
late StreamSubscription<BaseDevice> _connectionStateSubscription;
final controller = TextEditingController(text: actionHandler.supportedApp?.name);
final _snackBarMessengerKey = GlobalKey<ScaffoldMessengerState>();
bool _showAutoRotationWarning = false;
bool _showMiuiWarning = false;
StreamSubscription<bool>? _autoRotateStream;
@override
void initState() {
super.initState();
// keep screen on - this is required for iOS to keep the bluetooth connection alive
WakelockPlus.enable();
WidgetsBinding.instance.addObserver(this);
WidgetsBinding.instance.addPostFrameCallback((_) {
_checkAndShowChangelog();
});
if (!kIsWeb) {
whooshLink.isStarted.addListener(() {
if (mounted) setState(() {});
});
zwiftEmulator.isConnected.addListener(() {
if (mounted) setState(() {});
});
if (settings.getZwiftEmulatorEnabled() && actionHandler.supportedApp?.supportsZwiftEmulation == true) {
zwiftEmulator.startAdvertising(() {
if (mounted) setState(() {});
});
}
}
if (actionHandler is RemoteActions && !kIsWeb && Platform.isIOS && (actionHandler as RemoteActions).isConnected) {
WidgetsBinding.instance.addPostFrameCallback((_) {
// show snackbar to inform user that the app needs to stay in foreground
_snackBarMessengerKey.currentState?.showSnackBar(
SnackBar(
content: Text('To simulate touches the app needs to stay in the foreground.'),
duration: Duration(seconds: 5),
),
);
});
}
_connectionStateSubscription = connection.connectionStream.listen((state) async {
_connectionStateSubscription = core.connection.connectionStream.listen((state) async {
setState(() {});
});
if (!kIsWeb && Platform.isAndroid) {
DeviceAutoRotateChecker.checkAutoRotate().then((isEnabled) {
if (!isEnabled) {
setState(() {
_showAutoRotationWarning = true;
});
}
});
_autoRotateStream = DeviceAutoRotateChecker.autoRotateStream.listen((isEnabled) {
setState(() {
_showAutoRotationWarning = !isEnabled;
});
});
// Check if device is MIUI and using local accessibility service
if (actionHandler is AndroidActions) {
_checkMiuiDevice();
}
}
}
@override
void dispose() {
WidgetsBinding.instance.removeObserver(this);
_autoRotateStream?.cancel();
_connectionStateSubscription.cancel();
controller.dispose();
super.dispose();
}
@override
void didChangeAppLifecycleState(AppLifecycleState state) {
if (state == AppLifecycleState.resumed) {
if (actionHandler is RemoteActions && Platform.isIOS && (actionHandler as RemoteActions).isConnected) {
UniversalBle.getBluetoothAvailabilityState().then((state) {
if (state == AvailabilityState.poweredOn) {
final requirement = RemoteRequirement();
requirement.reconnect();
_snackBarMessengerKey.currentState?.showSnackBar(
SnackBar(
content: Text('To simulate touches the app needs to stay in the foreground.'),
duration: Duration(seconds: 5),
),
);
}
});
}
}
}
Future<void> _checkMiuiDevice() async {
try {
// Don't show if user has dismissed the warning
if (settings.getMiuiWarningDismissed()) {
return;
}
final deviceInfo = await DeviceInfoPlugin().androidInfo;
final isMiui =
deviceInfo.manufacturer.toLowerCase() == 'xiaomi' ||
deviceInfo.brand.toLowerCase() == 'xiaomi' ||
deviceInfo.brand.toLowerCase() == 'redmi' ||
deviceInfo.brand.toLowerCase() == 'poco';
if (isMiui && mounted) {
setState(() {
_showMiuiWarning = true;
});
}
} catch (e) {
// Silently fail if device info is not available
}
}
Future<void> _checkAndShowChangelog() async {
try {
final packageInfo = await PackageInfo.fromPlatform();
final currentVersion = packageInfo.version;
final lastSeenVersion = settings.getLastSeenVersion();
if (mounted) {
await ChangelogDialog.showIfNeeded(context, currentVersion, lastSeenVersion);
}
// Update last seen version
await settings.setLastSeenVersion(currentVersion);
} catch (e) {
print('Failed to check changelog: $e');
}
}
@override
Widget build(BuildContext context) {
final canVibrate = connection.bluetoothDevices.any(
(device) => (device.device.name == 'Zwift Ride' || device.device.name == 'Zwift Play') && device.isConnected,
);
final paddingMultiplicator = actionHandler is DesktopActions ? 2.5 : 1.0;
return ScaffoldMessenger(
key: _snackBarMessengerKey,
child: PopScope(
onPopInvokedWithResult: (hello, _) {
connection.reset();
},
child: Stack(
return Scrollbar(
child: SingleChildScrollView(
primary: true,
padding: EdgeInsets.all(16),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
spacing: 12,
children: [
Scaffold(
appBar: AppBar(
title: AppTitle(),
actions: buildMenuButtons(),
backgroundColor: Theme.of(context).colorScheme.inversePrimary,
ValueListenableBuilder(
valueListenable: IAPManager.instance.isPurchased,
builder: (context, value, child) => value ? SizedBox.shrink() : IAPStatusWidget(small: false),
),
if (core.connection.controllerDevices.isEmpty)
Padding(
padding: const EdgeInsets.only(top: 8.0),
child: ColoredTitle(text: context.i18n.connectControllers),
),
body: SingleChildScrollView(
padding: EdgeInsets.only(
top: 16,
left: 8.0 * paddingMultiplicator,
right: 8 * paddingMultiplicator,
bottom: 8,
),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
if (_showAutoRotationWarning)
Warning(
children: [
Text('Enable auto-rotation on your device to make sure the app works correctly.'),
],
),
if (_showMiuiWarning)
Warning(
children: [
Row(
children: [
Icon(Icons.warning_amber, color: Theme.of(context).colorScheme.error),
SizedBox(width: 8),
Expanded(
child: Text(
'MIUI Device Detected',
style: TextStyle(
fontWeight: FontWeight.bold,
color: Theme.of(context).colorScheme.error,
),
),
),
IconButton(
icon: Icon(Icons.close),
onPressed: () async {
await settings.setMiuiWarningDismissed(true);
setState(() {
_showMiuiWarning = false;
});
},
tooltip: 'Dismiss',
padding: EdgeInsets.zero,
constraints: BoxConstraints(),
),
],
),
SizedBox(height: 8),
Text(
'Your device is running MIUI, which is known to aggressively kill background services and accessibility services.',
style: TextStyle(fontSize: 14),
),
SizedBox(height: 8),
Text(
'To ensure SwiftControl works properly:',
style: TextStyle(fontSize: 14, fontWeight: FontWeight.w600),
),
Text(
'• Disable battery optimization for SwiftControl',
style: TextStyle(fontSize: 14),
),
Text(
'• Enable autostart for SwiftControl',
style: TextStyle(fontSize: 14),
),
Text(
'• Lock the app in recent apps',
style: TextStyle(fontSize: 14),
),
SizedBox(height: 12),
ElevatedButton.icon(
onPressed: () async {
final url = Uri.parse('https://dontkillmyapp.com/xiaomi');
if (await canLaunchUrl(url)) {
await launchUrl(url, mode: LaunchMode.externalApplication);
}
},
icon: Icon(Icons.open_in_new),
label: Text('View Detailed Instructions'),
),
],
),
Padding(
padding: const EdgeInsets.symmetric(horizontal: 4),
child: Text('Connected Devices', style: Theme.of(context).textTheme.titleMedium),
),
Card(
child: Padding(
padding: EdgeInsets.only(
left: 16.0,
right: 16,
top: 16,
bottom: actionHandler is RemoteActions ? 0 : 12,
),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Container(
margin: const EdgeInsets.only(bottom: 8.0),
width: double.infinity,
decoration: BoxDecoration(
border: Border(
bottom: BorderSide(
color: Theme.of(context).colorScheme.primaryContainer,
),
),
),
child: Padding(
padding: const EdgeInsets.symmetric(vertical: 8.0),
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Text(
'Connected Controllers',
style: TextStyle(fontWeight: FontWeight.bold),
),
if (connection.controllerDevices.isEmpty) SmallProgressIndicator(),
],
),
),
),
if (connection.controllerDevices.isEmpty)
ScanWidget()
else
...connection.controllerDevices.map(
(device) => device.showInformation(context),
),
if (connection.remoteDevices.isNotEmpty ||
actionHandler is RemoteActions ||
whooshLink.isCompatible(settings.getLastTarget() ?? Target.thisDevice) ||
actionHandler.supportedApp?.supportsZwiftEmulation == true)
Container(
margin: const EdgeInsets.only(bottom: 8.0),
width: double.infinity,
decoration: BoxDecoration(
border: Border(
bottom: BorderSide(
color: Theme.of(context).colorScheme.primaryContainer,
),
),
),
child: Padding(
padding: const EdgeInsets.symmetric(vertical: 8.0),
child: Text(
'Remote Connections',
style: TextStyle(fontWeight: FontWeight.bold),
),
),
),
...connection.remoteDevices.map(
(device) => device.showInformation(context),
),
// leave it in for the extra scanning options
ScanWidget(),
if (settings.getTrainerApp() is MyWhoosh &&
whooshLink.isCompatible(settings.getLastTarget()!))
LinkDevice('').showInformation(context),
if (settings.getTrainerApp()?.supportsZwiftEmulation == true)
ZwiftRequirement().build(context, () {
setState(() {});
})!,
if (core.connection.controllerDevices.isNotEmpty)
Padding(
padding: const EdgeInsets.only(top: 8.0),
child: ColoredTitle(text: context.i18n.connectedControllers),
),
if (actionHandler is RemoteActions)
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Text(
'Remote Control Mode: ${(actionHandler as RemoteActions).isConnected ? 'Connected' : 'Not connected'}',
),
PopupMenuButton(
itemBuilder: (_) => [
PopupMenuItem(
child: Text('Reconnect'),
onTap: () async {
final requirement = RemoteRequirement();
await requirement.reconnect();
},
),
],
),
],
),
],
),
),
),
SizedBox(height: 20),
if (!kIsWeb) ...[
Padding(
padding: const EdgeInsets.symmetric(horizontal: 4),
child: Text(
'Customize ${settings.getTrainerApp()?.name} on ${settings.getLastTarget()?.title}',
style: Theme.of(context).textTheme.titleMedium,
),
),
Card(
child: Padding(
padding: EdgeInsets.only(
left: 16.0,
right: 16,
top: 16,
bottom: canVibrate ? 0 : 12,
),
child: Column(
spacing: 12,
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
spacing: 8,
children: [
Expanded(
child: DropdownMenu<SupportedApp?>(
controller: controller,
dropdownMenuEntries: [
..._getAllApps().map(
(app) => DropdownMenuEntry<SupportedApp>(
value: app,
label: app.name,
labelWidget: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Text(app.name),
if (app is CustomApp) BetaPill(text: 'CUSTOM'),
],
),
),
),
DropdownMenuEntry(
value: CustomApp(profileName: 'New'),
label: 'Create new keymap',
labelWidget: Text('Create new keymap'),
leadingIcon: Icon(Icons.add),
),
],
label: Text('Select Keymap'),
onSelected: (app) async {
if (app == null) {
return;
} else if (app.name == 'New') {
final profileName = await KeymapManager().showNewProfileDialog(context);
if (profileName != null && profileName.isNotEmpty) {
final customApp = CustomApp(profileName: profileName);
actionHandler.init(customApp);
await settings.setKeyMap(customApp);
controller.text = profileName;
setState(() {});
}
} else {
controller.text = app.name ?? '';
actionHandler.supportedApp = app;
await settings.setKeyMap(app);
setState(() {});
}
},
initialSelection: actionHandler.supportedApp,
hintText: 'Select your Keymap',
),
),
Row(
children: [
KeymapManager().getManageProfileDialog(
context,
actionHandler.supportedApp is CustomApp
? actionHandler.supportedApp?.name
: null,
onDone: () {
setState(() {});
controller.text = actionHandler.supportedApp?.name ?? '';
},
),
],
),
],
),
if (actionHandler.supportedApp is! CustomApp)
Text(
'Customize the keymap if you experience any issues (e.g. wrong keyboard output, or misaligned touch placements)',
style: TextStyle(fontSize: 12),
),
if (actionHandler.supportedApp != null && connection.controllerDevices.isNotEmpty)
KeymapExplanation(
key: Key(actionHandler.supportedApp!.keymap.runtimeType.toString()),
keymap: actionHandler.supportedApp!.keymap,
onUpdate: () {
setState(() {});
controller.text = actionHandler.supportedApp?.name ?? '';
if (actionHandler.supportedApp is CustomApp) {
settings.setKeyMap(actionHandler.supportedApp!);
}
},
),
if (canVibrate) ...[
SwitchListTile(
title: Text('Enable vibration feedback when shifting gears'),
value: settings.getVibrationEnabled(),
contentPadding: EdgeInsets.zero,
onChanged: (value) async {
await settings.setVibrationEnabled(value);
setState(() {});
},
),
],
],
),
),
),
],
SizedBox(height: 20),
Padding(
padding: const EdgeInsets.symmetric(horizontal: 4),
child: Text('Logs', style: Theme.of(context).textTheme.titleMedium),
),
LogViewer(),
],
),
...core.connection.controllerDevices.map(
(device) => Card(
filled: true,
fillColor: Theme.of(context).brightness == Brightness.dark
? Theme.of(context).colorScheme.card
: Theme.of(context).colorScheme.card.withLuminance(0.95),
child: device.showInformation(context),
),
),
Positioned.fill(child: Testbed()),
if (core.connection.accessories.isNotEmpty) ...[
Padding(
padding: const EdgeInsets.only(top: 8.0),
child: ColoredTitle(text: AppLocalizations.of(context).accessories),
),
...core.connection.accessories.map(
(device) => Card(
filled: true,
fillColor: Theme.of(context).brightness == Brightness.dark
? Theme.of(context).colorScheme.card
: Theme.of(context).colorScheme.card.withLuminance(0.95),
child: device.showInformation(context),
),
),
],
SizedBox(),
if (core.connection.controllerDevices.isNotEmpty)
Row(
spacing: 8,
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
PrimaryButton(
child: Text(context.i18n.connectToTrainerApp),
onPressed: () {
widget.onUpdate();
},
),
],
),
],
),
),
);
}
List<SupportedApp> _getAllApps() {
final baseApp = settings.getTrainerApp();
final customProfiles = settings.getCustomAppProfiles();
final customApps = customProfiles.map((profile) {
final customApp = CustomApp(profileName: profile);
final savedKeymap = settings.getCustomAppKeymap(profile);
if (savedKeymap != null) {
customApp.decodeKeymap(savedKeymap);
}
return customApp;
}).toList();
// If no custom profiles exist, add the default "Custom" one
if (customApps.isEmpty) {
customApps.add(CustomApp());
}
return [if (baseApp != null) baseApp, ...customApps];
}
}
extension Screenshot on String {

View File

@@ -1,8 +1,10 @@
import 'package:flex_color_scheme/flex_color_scheme.dart';
import 'package:flutter/material.dart';
import 'package:dartx/dartx.dart';
import 'package:flutter/material.dart' show BackButton;
import 'package:flutter/services.dart';
import 'package:flutter_md/flutter_md.dart';
import 'package:http/http.dart' as http;
import 'package:shadcn_flutter/shadcn_flutter.dart';
import 'package:bike_control/widgets/ui/colored_title.dart';
import 'package:url_launcher/url_launcher_string.dart';
class MarkdownPage extends StatefulWidget {
@@ -14,7 +16,7 @@ class MarkdownPage extends StatefulWidget {
}
class _ChangelogPageState extends State<MarkdownPage> {
Markdown? _markdown;
List<_Group>? _groups;
String? _error;
@override
@@ -26,58 +28,99 @@ class _ChangelogPageState extends State<MarkdownPage> {
Future<void> _loadChangelog() async {
try {
final md = await rootBundle.loadString(widget.assetPath);
setState(() {
_markdown = Markdown.fromString(md);
});
// load latest version
final response = await http.get(
Uri.parse('https://raw.githubusercontent.com/jonasbark/swiftcontrol/refs/heads/main/${widget.assetPath}'),
);
if (response.statusCode == 200) {
final latestMd = response.body;
if (latestMd != md) {
setState(() {
_markdown = Markdown.fromString(md);
});
}
}
_parseMarkdown(md);
} catch (e) {
setState(() {
_error = 'Failed to load changelog: $e';
});
} finally {
_loadOnlineVersion();
}
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text(widget.assetPath.replaceAll('.md', '').toLowerCase().capitalize),
backgroundColor: Theme.of(context).colorScheme.inversePrimary,
),
body: _error != null
headers: [
AppBar(
leading: [
BackButton(),
],
title: Text(
widget.assetPath
.replaceAll('.md', '')
.split('_')
.joinToString(separator: ' ', transform: (s) => s.toLowerCase().capitalize()),
),
),
],
child: _error != null
? Center(child: Text(_error!))
: _markdown == null
: _groups == null
? Center(child: CircularProgressIndicator())
: SingleChildScrollView(
padding: EdgeInsets.all(16),
child: Row(
children: [
Expanded(
child: MarkdownWidget(
markdown: _markdown!,
theme: MarkdownThemeData(
textStyle: TextStyle(fontSize: 14.0, color: Colors.black87),
onLinkTap: (title, url) {
launchUrlString(url);
},
child: Accordion(
items: _groups!
.map(
(group) => AccordionItem(
trigger: AccordionTrigger(child: ColoredTitle(text: group.title)),
content: MarkdownWidget(
markdown: group.markdown,
theme: MarkdownThemeData(
textStyle: TextStyle(
fontSize: 14.0,
color: Theme.of(context).colorScheme.brightness == Brightness.dark
? Colors.white.withAlpha(255 * 70)
: Colors.black.withAlpha(87 * 255),
),
onLinkTap: (title, url) {
launchUrlString(url);
},
),
),
),
),
),
],
)
.toList(),
),
),
);
}
void _parseMarkdown(String md) {
setState(() {
_error = null;
_groups = md
.split('## ')
.map((section) {
final lines = section.split('\n');
final title = lines.first.replaceFirst('# ', '').trim();
final content = lines.skip(1).join('\n').trim();
return _Group(
title: title,
markdown: Markdown.fromString('## $content'),
);
})
.where((group) => group.title.isNotEmpty)
.toList();
});
}
Future<void> _loadOnlineVersion() async {
// load latest version
final response = await http.get(
Uri.parse('https://raw.githubusercontent.com/jonasbark/swiftcontrol/refs/heads/main/${widget.assetPath}'),
);
if (response.statusCode == 200) {
final latestMd = response.body;
_parseMarkdown(latestMd);
}
}
}
class _Group {
final String title;
final Markdown markdown;
_Group({required this.title, required this.markdown});
}

411
lib/pages/navigation.dart Normal file
View File

@@ -0,0 +1,411 @@
import 'dart:io';
import 'package:bike_control/gen/l10n.dart';
import 'package:bike_control/main.dart';
import 'package:bike_control/pages/customize.dart';
import 'package:bike_control/pages/device.dart';
import 'package:bike_control/pages/trainer.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/i18n_extension.dart';
import 'package:bike_control/widgets/logviewer.dart';
import 'package:bike_control/widgets/menu.dart';
import 'package:bike_control/widgets/title.dart';
import 'package:bike_control/widgets/ui/colors.dart';
import 'package:dartx/dartx.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/services.dart';
import 'package:package_info_plus/package_info_plus.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
import '../widgets/changelog_dialog.dart';
enum BCPage {
devices(Icons.gamepad),
trainer(Icons.pedal_bike),
customization(Icons.videogame_asset_outlined),
logs(Icons.article);
final IconData icon;
const BCPage(this.icon);
String getTitle(BuildContext context) {
return switch (this) {
BCPage.devices => context.i18n.controllers,
BCPage.trainer => context.i18n.trainer,
BCPage.customization => context.i18n.configuration,
BCPage.logs => context.i18n.logs,
};
}
}
class Navigation extends StatefulWidget {
final BCPage page;
const Navigation({super.key, this.page = BCPage.devices});
@override
State<Navigation> createState() => _NavigationState();
}
class _NavigationState extends State<Navigation> {
bool _isMobile = false;
late BCPage _selectedPage;
@override
void initState() {
super.initState();
_selectedPage = widget.page;
core.connection.initialize();
core.logic.startEnabledConnectionMethod();
core.connection.actionStream.listen((_) {
_updateTrainerConnectionStatus();
setState(() {});
});
_updateTrainerConnectionStatus();
WidgetsBinding.instance.addPostFrameCallback((_) {
SystemChrome.setSystemUIOverlayStyle(
Theme.of(context).colorScheme.brightness == Brightness.light
? SystemUiOverlayStyle.dark
: SystemUiOverlayStyle.light,
);
_checkAndShowChangelog();
});
}
@override
void didUpdateWidget(covariant Navigation oldWidget) {
super.didUpdateWidget(oldWidget);
if (widget.page != oldWidget.page) {
setState(() {
_selectedPage = widget.page;
});
}
}
void _updateTrainerConnectionStatus() async {
final isConnected = await core.logic.isTrainerConnected();
if (mounted) {
setState(() {
_isTrainerConnected = isConnected;
});
}
}
@override
void didChangeDependencies() {
super.didChangeDependencies();
_isMobile = MediaQuery.sizeOf(context).width < 600;
}
Future<void> _checkAndShowChangelog() async {
try {
final packageInfo = await PackageInfo.fromPlatform();
final currentVersion = packageInfo.version;
final lastSeenVersion = core.settings.getLastSeenVersion();
if (mounted) {
await ChangelogDialog.showIfNeeded(context, currentVersion, lastSeenVersion);
}
// Update last seen version
await core.settings.setLastSeenVersion(currentVersion);
} catch (e) {
print('Failed to check changelog: $e');
}
}
final List<BCPage> _tabs = BCPage.values.whereNot((e) => e == BCPage.logs).toList();
bool _isTrainerConnected = false;
@override
Widget build(BuildContext context) {
return Scaffold(
headers: [
AppBar(
padding:
const EdgeInsets.only(top: 12, bottom: 8, left: 12, right: 12) *
(screenshotMode ? 2 : Theme.of(context).scaling),
title: AppTitle(),
backgroundColor: Theme.of(context).colorScheme.background,
trailing: buildMenuButtons(
context,
_selectedPage,
_isMobile
? () {
setState(() {
_selectedPage = BCPage.logs;
});
}
: null,
),
),
Divider(),
],
footers: _isMobile ? [Divider(), _buildNavigationBar()] : [],
child: Row(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
if (!_isMobile) ...[
_buildNavigationMenu(),
VerticalDivider(),
],
Expanded(
child: Container(
alignment: Alignment.topLeft,
child: AnimatedSwitcher(
duration: Duration(milliseconds: 200),
child: switch (_selectedPage) {
BCPage.devices => DevicePage(
onUpdate: () {
setState(() {
_selectedPage = BCPage.trainer;
});
},
),
BCPage.trainer => TrainerPage(
onUpdate: () {
setState(() {});
},
goToNextPage: () {
setState(() {
_selectedPage = BCPage.customization;
});
},
),
BCPage.customization => CustomizePage(),
BCPage.logs => LogViewer(),
},
),
),
),
],
),
);
}
Widget _buildNavigationMenu() {
return Column(
children: [
Expanded(
child: NavigationSidebar(
backgroundColor: Theme.of(context).brightness == Brightness.light
? BKColor.backgroundLight
: Theme.of(context).colorScheme.card,
onSelected: (int index) {
setState(() {
_selectedPage = BCPage.values[index];
});
},
spacing: 4,
children: _tabs.map((page) => _buildNavigationItemDesktop(page)).toList(),
),
),
NavigationSidebar(
backgroundColor: Theme.of(context).brightness == Brightness.light
? BKColor.backgroundLight
: Theme.of(context).colorScheme.card,
onSelected: (int index) {
setState(() {
_selectedPage = BCPage.logs;
});
},
children: [
NavigationDivider(),
NavigationItem(
label: Text(BCPage.logs.getTitle(context)),
selected: _selectedPage == BCPage.logs,
child: _buildIcon(BCPage.logs),
),
],
),
],
);
}
Widget _buildIcon(BCPage page) {
final needsAttention = _needsAttention(page);
return Stack(
children: [
Icon(
page.icon,
color: !_isPageEnabled(page)
? null
: Theme.of(context).colorScheme.brightness == Brightness.dark
? Colors.white
: null,
),
if (needsAttention) ...[
Positioned(
right: 0,
top: 0,
child: RepeatedAnimationBuilder<double>(
duration: Duration(seconds: 1),
reverseDuration: Duration(seconds: 1),
start: 10,
end: 12,
mode: LoopingMode.pingPong,
builder: (context, value, child) {
return Container(
width: value,
height: value,
alignment: Alignment.center,
decoration: BoxDecoration(
color: Colors.red,
shape: BoxShape.circle,
border: Border.all(color: Colors.white, width: 1.5),
),
);
},
),
),
],
],
);
}
Widget _buildNavigationBar() {
return NavigationBar(
padding:
EdgeInsets.only(top: 6, left: 12, right: 12, bottom: !kIsWeb && Platform.isMacOS ? 8 : 0) *
Theme.of(context).scaling,
labelType: NavigationLabelType.all,
onSelected: (int index) {
setState(() {
_selectedPage = _tabs[index];
});
},
children: _tabs.map((page) {
return NavigationItem(
selected: _selectedPage == page,
selectedStyle: ButtonStyle.primary(density: ButtonDensity.dense).copyWith(
decoration: (context, states, value) {
return BoxDecoration(
gradient: const LinearGradient(
colors: [BKColor.main, BKColor.mainEnd],
),
borderRadius: BorderRadius.circular(8),
);
},
),
style: ButtonStyle.ghost(density: ButtonDensity.dense).copyWith(
decoration: (context, states, value) {
return BoxDecoration(
gradient: states.contains(WidgetState.hovered)
? const LinearGradient(
colors: [BKColor.main, BKColor.mainEnd],
)
: null,
borderRadius: BorderRadius.circular(8),
);
},
),
enabled: _isPageEnabled(page),
label: Text(
page == BCPage.trainer && !screenshotMode
? core.settings.getTrainerApp()?.name.split(' ').first ?? page.getTitle(context)
: page.getTitle(context),
style: TextStyle(
color: !_isPageEnabled(page)
? null
: Theme.of(context).colorScheme.brightness == Brightness.dark
? Colors.white
: null,
),
),
child: _buildIcon(page),
);
}).toList(),
);
}
bool _isPageEnabled(BCPage page) {
return switch (page) {
BCPage.customization => core.settings.getTrainerApp() != null,
_ => true,
};
}
bool _needsAttention(BCPage page) {
return switch (page) {
BCPage.devices => core.connection.controllerDevices.isEmpty,
BCPage.customization => false,
BCPage.trainer => core.settings.getTrainerApp() == null || !_isTrainerConnected,
BCPage.logs => false,
};
}
NavigationBarItem _buildNavigationItemDesktop(BCPage page) {
return NavigationItem(
selected: _selectedPage == page,
selectedStyle: ButtonStyle.primary(density: ButtonDensity.dense).copyWith(
decoration: (context, states, value) {
return BoxDecoration(
gradient: const LinearGradient(
colors: [BKColor.main, BKColor.mainEnd],
),
borderRadius: BorderRadius.circular(8),
);
},
padding: (context, states, value) {
return EdgeInsets.symmetric(horizontal: 12, vertical: 16);
},
),
style: ButtonStyle.ghost(density: ButtonDensity.dense).copyWith(
decoration: (context, states, value) {
return BoxDecoration(
gradient: states.contains(WidgetState.hovered)
? const LinearGradient(
colors: [BKColor.main, BKColor.mainEnd],
)
: null,
borderRadius: BorderRadius.circular(8),
);
},
padding: (context, states, value) {
return EdgeInsets.symmetric(horizontal: 12, vertical: 16);
},
),
enabled: _isPageEnabled(page),
child: SizedBox(
width: screenshotMode ? 180 : 152,
child: Basic(
padding: screenshotMode ? EdgeInsets.all(0) : null,
leading: _buildIcon(page),
leadingAlignment: Alignment.centerLeft,
title: Text(
page == BCPage.trainer && !screenshotMode
? core.settings.getTrainerApp()?.name.split(' ').first ?? page.getTitle(context)
: page.getTitle(context),
style: TextStyle(
color: !_isPageEnabled(page)
? null
: Theme.of(context).colorScheme.brightness == Brightness.dark
? Colors.white
: null,
),
),
subtitle: _needsAttention(page)
? Text(
switch (page) {
BCPage.devices => AppLocalizations.of(context).noControllerConnected,
BCPage.trainer when !_isTrainerConnected => AppLocalizations.of(context).notConnected,
BCPage.trainer when core.settings.getTrainerApp() == null => AppLocalizations.of(
context,
).noTrainerSelected,
_ => '',
},
style: _selectedPage == page ? TextStyle(color: Colors.gray.shade300) : null,
)
: null,
),
),
);
}
}

View File

@@ -1,207 +0,0 @@
import 'dart:io';
import 'package:dartx/dartx.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:swift_control/main.dart';
import 'package:swift_control/utils/requirements/multi.dart';
import 'package:swift_control/utils/requirements/platform.dart';
import 'package:swift_control/widgets/menu.dart';
import 'package:swift_control/widgets/small_progress_indicator.dart';
import 'package:swift_control/widgets/title.dart';
import 'device.dart';
class RequirementsPage extends StatefulWidget {
const RequirementsPage({super.key});
@override
State<RequirementsPage> createState() => _RequirementsPageState();
}
class _RequirementsPageState extends State<RequirementsPage> with WidgetsBindingObserver {
int _currentStep = 0;
List<PlatformRequirement> _requirements = [];
@override
void initState() {
super.initState();
WidgetsBinding.instance.addObserver(this);
// call after first frame
WidgetsBinding.instance.addPostFrameCallback((_) {
settings.init().then((_) {
if (!kIsWeb && Platform.isMacOS) {
// add more delay due to CBManagerStateUnknown
Future.delayed(const Duration(seconds: 2), () {
_reloadRequirements();
});
} else {
_reloadRequirements();
}
});
});
}
@override
dispose() {
WidgetsBinding.instance.removeObserver(this);
super.dispose();
}
@override
void didChangeAppLifecycleState(AppLifecycleState state) {
if (state == AppLifecycleState.resumed) {
_reloadRequirements();
}
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: AppTitle(),
backgroundColor: Theme.of(context).colorScheme.inversePrimary,
actions: buildMenuButtons(),
),
body: SingleChildScrollView(
child: Column(
spacing: 12,
children: [
SizedBox(height: 12),
Row(
mainAxisSize: MainAxisSize.min,
spacing: 12,
children: [
Image.asset('icon.png', width: 64, height: 64),
Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text('Welcome to SwiftControl!', style: Theme.of(context).textTheme.titleMedium),
Container(
constraints: BoxConstraints(maxWidth: MediaQuery.sizeOf(context).width - 140),
child: Text.rich(
TextSpan(
children: [
TextSpan(text: 'Need help? Click on the '),
WidgetSpan(
child: Padding(
padding: const EdgeInsets.only(top: 4.0),
child: Icon(Icons.help_outline),
),
),
TextSpan(text: ' button on top and don\'t hesitate to contact us.'),
],
),
),
),
],
),
],
),
_requirements.isEmpty
? Center(child: SmallProgressIndicator())
: Card(
margin: EdgeInsets.symmetric(horizontal: 16, vertical: 16),
child: Stepper(
physics: NeverScrollableScrollPhysics(),
currentStep: _currentStep,
connectorColor: WidgetStateProperty.resolveWith<Color>(
(Set<WidgetState> states) => Theme.of(context).colorScheme.primary,
),
onStepContinue: _currentStep < _requirements.length
? () {
setState(() {
_currentStep += 1;
});
}
: null,
onStepTapped: (step) {
if (_requirements[step].status && _requirements[step] is! TargetRequirement) {
return;
}
final hasEarlierIncomplete =
_requirements.indexWhere((req) => !req.status) != -1 &&
_requirements.indexWhere((req) => !req.status) < step;
if (hasEarlierIncomplete) {
return;
}
setState(() {
_currentStep = step;
});
},
controlsBuilder: (context, details) => Container(),
steps: _requirements
.mapIndexed(
(index, req) => Step(
title: Text(req.name, style: TextStyle(fontWeight: FontWeight.w600)),
subtitle:
req.buildDescription() ?? (req.description != null ? Text(req.description!) : null),
content: Container(
padding: const EdgeInsets.only(top: 16.0),
alignment: Alignment.centerLeft,
child:
(index == _currentStep
? req.build(context, () {
_reloadRequirements();
})
: null) ??
ElevatedButton(
onPressed: req.status
? null
: () => _callRequirement(req, context, () {
_reloadRequirements();
}),
child: Text(req.name),
),
),
state: req.status ? StepState.complete : StepState.indexed,
),
)
.toList(),
),
),
],
),
),
);
}
void _callRequirement(PlatformRequirement req, BuildContext context, VoidCallback onUpdate) {
req.call(context, onUpdate).then((_) {
_reloadRequirements();
});
}
void _reloadRequirements() {
getRequirements(
settings.getLastTarget()?.connectionType ?? ConnectionType.unknown,
).then((req) {
_requirements = req;
final unresolvedIndex = req.indexWhere((req) => !req.status);
if (unresolvedIndex != -1) {
_currentStep = unresolvedIndex;
} else if (mounted) {
String? currentPath;
navigatorKey.currentState?.popUntil((route) {
currentPath = route.settings.name;
return true;
});
if (currentPath == '/') {
Navigator.push(
context,
MaterialPageRoute(
builder: (c) => DevicePage(),
settings: RouteSettings(name: '/device'),
),
);
}
}
if (mounted) {
setState(() {});
}
});
}
}

View File

@@ -2,21 +2,20 @@ import 'dart:async';
import 'dart:io';
import 'dart:math';
import 'package:dartx/dartx.dart';
import 'package:bike_control/main.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/i18n_extension.dart';
import 'package:bike_control/widgets/keymap_explanation.dart';
import 'package:bike_control/widgets/testbed.dart';
import 'package:bike_control/widgets/ui/button_widget.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter/scheduler.dart';
import 'package:flutter/services.dart';
import 'package:image_picker/image_picker.dart';
import 'package:path_provider/path_provider.dart';
import 'package:swift_control/main.dart';
import 'package:swift_control/utils/keymap/apps/my_whoosh.dart';
import 'package:swift_control/widgets/button_widget.dart';
import 'package:swift_control/widgets/keymap_explanation.dart';
import 'package:swift_control/widgets/testbed.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
import 'package:window_manager/window_manager.dart';
import '../utils/actions/base_actions.dart';
import '../utils/keymap/keymap.dart';
final touchAreaSize = 42.0;
@@ -43,7 +42,7 @@ class _TouchAreaSetupPageState extends State<TouchAreaSetupPage> {
if (result != null) {
final image = File(result.path);
final Directory tempDir = await getTemporaryDirectory();
final tempImage = File('${tempDir.path}/${actionHandler.supportedApp?.name ?? 'temp'}_screenshot.png');
final tempImage = File('${tempDir.path}/${core.actionHandler.supportedApp?.name ?? 'temp'}_screenshot.png');
await image.copy(tempImage.path);
_backgroundImage = tempImage.readAsBytesSync();
await _calculateBounds();
@@ -113,7 +112,7 @@ class _TouchAreaSetupPageState extends State<TouchAreaSetupPage> {
windowManager.setFullScreen(true);
}
getTemporaryDirectory().then((tempDir) async {
final tempImage = File('${tempDir.path}/${actionHandler.supportedApp?.name ?? 'temp'}_screenshot.png');
final tempImage = File('${tempDir.path}/${core.actionHandler.supportedApp?.name ?? 'temp'}_screenshot.png');
if (tempImage.existsSync()) {
_backgroundImage = tempImage.readAsBytesSync();
setState(() {});
@@ -197,7 +196,7 @@ class _TouchAreaSetupPageState extends State<TouchAreaSetupPage> {
left: position.dx,
top: position.dy,
child: Tooltip(
message: 'Drag to reposition',
tooltip: (c) => Text(context.i18n.dragToReposition),
child: AnimatedOpacity(
opacity: _showFaded && widget.keyPair != keyPair ? 0.2 : 1.0,
duration: Duration(milliseconds: 300),
@@ -207,7 +206,7 @@ class _TouchAreaSetupPageState extends State<TouchAreaSetupPage> {
final RenderBox renderObject = context.findRenderObject() as RenderBox;
return renderObject.globalToLocal(position).scale(scale, scale);
},
feedback: Material(
feedback: Container(
color: Colors.transparent,
child: icon,
),
@@ -250,13 +249,13 @@ class _TouchAreaSetupPageState extends State<TouchAreaSetupPage> {
@override
Widget build(BuildContext context) {
return Scaffold(
body: LayoutBuilder(
child: LayoutBuilder(
builder: (context, constraints) {
if (_backgroundImage == null && constraints.biggest != _imageRect.size) {
_imageRect = Rect.fromLTWH(0, 0, constraints.maxWidth, constraints.maxHeight);
}
final keyPairsToShow =
actionHandler.supportedApp?.keymap.keyPairs
core.actionHandler.supportedApp?.keymap.keyPairs
.where((kp) => kp.touchPosition != Offset.zero && !kp.isSpecialKey)
.toList() ??
[];
@@ -314,19 +313,14 @@ class _TouchAreaSetupPageState extends State<TouchAreaSetupPage> {
children: [
IgnorePointer(
child: Text(
'''1. Create an in-game screenshot of your app (e.g. within MyWhoosh) in landscape orientation
2. Load the screenshot with the button below
3. The app is automatically set to landscape orientation for accurate mapping
4. Press a button on your Click device to create a touch area
5. Drag the touch areas to the desired position on the screenshot
6. Save and close this screen''',
context.i18n.touchAreaInstructions,
),
),
ElevatedButton(
PrimaryButton(
onPressed: () {
_pickScreenshot();
},
child: Text('Load in-game screenshot for placement'),
child: Text(context.i18n.loadScreenshotForPlacement),
),
],
),
@@ -339,30 +333,42 @@ class _TouchAreaSetupPageState extends State<TouchAreaSetupPage> {
child: Row(
spacing: 8,
children: [
ElevatedButton.icon(
IconButton.outline(
onPressed: _saveAndClose,
icon: const Icon(Icons.save),
label: const Text("Save"),
trailing: Text(context.i18n.save),
),
PopupMenuButton(
itemBuilder: (c) => [
PopupMenuItem(
child: Text('Choose another screenshot'),
onTap: () {
_pickScreenshot();
},
),
PopupMenuItem(
child: Text('Reset'),
onTap: () {
_backgroundImage = null;
Builder(
builder: (context) {
return OutlineButton(
child: Text('Menu'),
onPressed: () {
showDropdown(
context: context,
builder: (c) => DropdownMenu(
children: [
if (_backgroundImage != null)
MenuButton(
child: Text(context.i18n.chooseAnotherScreenshot),
onPressed: (c) {
_pickScreenshot();
},
),
MenuButton(
child: Text(context.i18n.reset),
onPressed: (c) {
_backgroundImage = null;
actionHandler.supportedApp?.keymap.reset();
setState(() {});
core.actionHandler.supportedApp?.keymap.reset();
setState(() {});
},
),
],
),
);
},
),
],
icon: Icon(Icons.more_vert),
);
},
),
],
),
@@ -384,59 +390,24 @@ class KeypairExplanation extends StatelessWidget {
@override
Widget build(BuildContext context) {
return Wrap(
spacing: 4,
runSpacing: 4,
crossAxisAlignment: WrapCrossAlignment.center,
children: [
if (withKey)
Row(
children: keyPair.buttons.map((b) => ButtonWidget(button: b, big: true)).toList(),
)
else
Icon(keyPair.icon),
if (keyPair.inGameAction != null &&
((settings.getTrainerApp() is MyWhoosh && settings.getMyWhooshLinkEnabled()) ||
(settings.getTrainerApp()?.supportsZwiftEmulation == true && settings.getZwiftEmulatorEnabled())))
_KeyWidget(
label: [
keyPair.inGameAction.toString().split('.').last,
if (keyPair.inGameActionValue != null) ': ${keyPair.inGameActionValue}',
].joinToString(separator: ''),
)
else if (keyPair.isSpecialKey && actionHandler.supportedModes.contains(SupportedMode.media))
_KeyWidget(
label: switch (keyPair.physicalKey) {
PhysicalKeyboardKey.mediaPlayPause => 'Play/Pause',
PhysicalKeyboardKey.mediaStop => 'Stop',
PhysicalKeyboardKey.mediaTrackPrevious => 'Previous',
PhysicalKeyboardKey.mediaTrackNext => 'Next',
PhysicalKeyboardKey.audioVolumeUp => 'Volume Up',
PhysicalKeyboardKey.audioVolumeDown => 'Volume Down',
_ => 'Unknown',
},
)
else if (keyPair.physicalKey != null && actionHandler.supportedModes.contains(SupportedMode.keyboard)) ...[
_KeyWidget(
label: [
...keyPair.modifiers.map((e) => e.name.replaceAll('Modifier', '')),
keyPair.logicalKey?.keyLabel ?? 'Unknown',
].joinToString(separator: '+'),
),
if (keyPair.isLongPress) Text('long\npress', style: TextStyle(fontSize: 10)),
] else ...[
if (!withKey && keyPair.touchPosition != Offset.zero)
_KeyWidget(label: 'X:${keyPair.touchPosition.dx.toInt()}, Y:${keyPair.touchPosition.dy.toInt()}'),
if (keyPair.isLongPress) Text('long\npress', style: TextStyle(fontSize: 10)),
],
],
return Basic(
leading: withKey
? Row(
children: keyPair.buttons.map((b) => ButtonWidget(button: b, big: true)).toList(),
)
: Icon(keyPair.icon),
leadingAlignment: Alignment.centerLeft,
contentSpacing: 10,
subtitle: keyPair.isLongPress ? Text(context.i18n.longPress.replaceAll('\n', ' ')).muted.xSmall : null,
title: Text(keyPair.toString()),
);
}
}
class _KeyWidget extends StatelessWidget {
class KeyWidget extends StatelessWidget {
final String label;
const _KeyWidget({super.key, required this.label});
final bool invert;
const KeyWidget({super.key, required this.label, this.invert = false});
@override
Widget build(BuildContext context) {
@@ -445,17 +416,17 @@ class _KeyWidget extends StatelessWidget {
padding: const EdgeInsets.symmetric(horizontal: 4, vertical: 4),
constraints: BoxConstraints(minWidth: 30),
decoration: BoxDecoration(
border: Border.all(color: Theme.of(context).colorScheme.primary),
color: invert ? Colors.white : Colors.black,
border: Border.all(color: Theme.of(context).colorScheme.border, width: 2),
borderRadius: BorderRadius.circular(4),
color: Theme.of(context).colorScheme.primaryContainer,
),
child: Center(
child: Text(
label.splitByUpperCase(),
style: TextStyle(
fontFamily: 'monospace',
fontFamily: screenshotMode ? null : 'monospace',
fontSize: 12,
color: Theme.of(context).colorScheme.onPrimaryContainer,
color: invert ? Colors.black : Colors.white,
),
),
),

236
lib/pages/trainer.dart Normal file
View File

@@ -0,0 +1,236 @@
import 'package:bike_control/bluetooth/messages/notification.dart';
import 'package:bike_control/gen/l10n.dart';
import 'package:bike_control/main.dart';
import 'package:bike_control/pages/button_simulator.dart';
import 'package:bike_control/pages/configuration.dart';
import 'package:bike_control/pages/navigation.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/i18n_extension.dart';
import 'package:bike_control/utils/iap/iap_manager.dart';
import 'package:bike_control/utils/requirements/multi.dart';
import 'package:bike_control/widgets/apps/local_tile.dart';
import 'package:bike_control/widgets/apps/mywhoosh_link_tile.dart';
import 'package:bike_control/widgets/apps/openbikecontrol_ble_tile.dart';
import 'package:bike_control/widgets/apps/openbikecontrol_mdns_tile.dart';
import 'package:bike_control/widgets/apps/zwift_mdns_tile.dart';
import 'package:bike_control/widgets/apps/zwift_tile.dart';
import 'package:bike_control/widgets/iap_status_widget.dart';
import 'package:bike_control/widgets/pair_widget.dart';
import 'package:bike_control/widgets/ui/colored_title.dart';
import 'package:bike_control/widgets/ui/toast.dart';
import 'package:flutter/foundation.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
import 'package:universal_ble/universal_ble.dart';
import 'package:wakelock_plus/wakelock_plus.dart';
import '../bluetooth/devices/zwift/protocol/zp.pbenum.dart';
class TrainerPage extends StatefulWidget {
final VoidCallback onUpdate;
final VoidCallback goToNextPage;
const TrainerPage({super.key, required this.onUpdate, required this.goToNextPage});
@override
State<TrainerPage> createState() => _TrainerPageState();
}
class _TrainerPageState extends State<TrainerPage> with WidgetsBindingObserver {
late final ScrollController _scrollController = ScrollController();
@override
void initState() {
super.initState();
WidgetsBinding.instance.addObserver(this);
// keep screen on - this is required for iOS to keep the bluetooth connection alive
if (!screenshotMode) {
WakelockPlus.enable();
}
if (!kIsWeb) {
if (core.logic.showForegroundMessage) {
WidgetsBinding.instance.addPostFrameCallback((_) {
// show snackbar to inform user that the app needs to stay in foreground
buildToast(context, title: AppLocalizations.current.touchSimulationForegroundMessage);
});
}
core.whooshLink.isStarted.addListener(() {
if (mounted) setState(() {});
});
core.zwiftEmulator.isConnected.addListener(() {
if (mounted) setState(() {});
});
}
}
@override
void dispose() {
WidgetsBinding.instance.removeObserver(this);
_scrollController.dispose();
super.dispose();
}
@override
void didChangeAppLifecycleState(AppLifecycleState state) {
if (state == AppLifecycleState.resumed) {
if (core.logic.showForegroundMessage) {
UniversalBle.getBluetoothAvailabilityState().then((state) {
if (state == AvailabilityState.poweredOn && mounted) {
core.remotePairing.reconnect();
buildToast(context, title: AppLocalizations.current.touchSimulationForegroundMessage);
}
});
}
}
}
@override
Widget build(BuildContext context) {
final showLocalAsOther =
(core.logic.showObpBluetoothEmulator || core.logic.showObpMdnsEmulator) && core.logic.showLocalControl;
final showWhooshLinkAsOther =
(core.logic.showObpBluetoothEmulator || core.logic.showObpMdnsEmulator) && core.logic.showMyWhooshLink;
final isMobile = MediaQuery.sizeOf(context).width < 800;
return Scrollbar(
controller: _scrollController,
child: SingleChildScrollView(
controller: _scrollController,
padding: EdgeInsets.all(16),
child: Column(
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
spacing: 12,
children: [
ValueListenableBuilder(
valueListenable: IAPManager.instance.isPurchased,
builder: (context, value, child) => value ? SizedBox.shrink() : IAPStatusWidget(small: true),
),
ConfigurationPage(
onUpdate: () {
setState(() {});
widget.onUpdate();
if (_scrollController.position.pixels != _scrollController.position.maxScrollExtent &&
core.settings.getLastTarget() == Target.otherDevice) {
WidgetsBinding.instance.addPostFrameCallback((_) {
_scrollController.animateTo(
_scrollController.offset + 300,
duration: Duration(milliseconds: 300),
curve: Curves.easeInOut,
);
});
}
},
),
if (core.settings.getTrainerApp() != null) ...[
SizedBox(height: 8),
if (core.logic.hasRecommendedConnectionMethods)
ColoredTitle(text: context.i18n.recommendedConnectionMethods),
if (core.logic.showObpMdnsEmulator) OpenBikeControlMdnsTile(),
if (core.logic.showObpBluetoothEmulator) OpenBikeControlBluetoothTile(),
if (core.logic.showZwiftMsdnEmulator)
ZwiftMdnsTile(
onUpdate: () {
core.connection.signalNotification(
LogNotification('Zwift Emulator status changed to ${core.zwiftEmulator.isConnected.value}'),
);
},
),
if (core.logic.showZwiftBleEmulator)
ZwiftTile(
onUpdate: () {
if (mounted) {
core.connection.signalNotification(
LogNotification('Zwift Emulator status changed to ${core.zwiftEmulator.isConnected.value}'),
);
setState(() {});
}
},
),
if (core.logic.showLocalControl && !showLocalAsOther) LocalTile(),
if (core.logic.showMyWhooshLink && !showWhooshLinkAsOther) MyWhooshLinkTile(),
Text.rich(
TextSpan(
children: [
TextSpan(text: '${context.i18n.needHelpClickHelp} '),
WidgetSpan(
child: Padding(
padding: const EdgeInsets.only(top: 4.0),
child: Icon(Icons.help_outline),
),
),
TextSpan(text: ' ${context.i18n.needHelpDontHesitate}'),
],
),
).small.muted,
if (core.logic.showRemote || showLocalAsOther || showWhooshLinkAsOther) ...[
SizedBox(height: 16),
Accordion(
items: [
AccordionItem(
trigger: AccordionTrigger(child: ColoredTitle(text: context.i18n.otherConnectionMethods)),
content: Column(
children: [
if (core.logic.showRemote) RemotePairingWidget(),
if (showLocalAsOther) LocalTile(),
if (showWhooshLinkAsOther) MyWhooshLinkTile(),
],
),
),
],
),
],
SizedBox(height: 4),
Flex(
direction: isMobile ? Axis.vertical : Axis.horizontal,
mainAxisAlignment: MainAxisAlignment.spaceBetween,
crossAxisAlignment: CrossAxisAlignment.start,
spacing: 8,
children: [
PrimaryButton(
leading: Icon(Icons.computer_outlined),
child: Text(
AppLocalizations.of(
context,
).manualyControllingButton(core.settings.getTrainerApp()?.name ?? 'your trainer'),
),
onPressed: () {
if (core.settings.getTrainerApp() == null) {
buildToast(
context,
level: LogLevel.LOGLEVEL_WARNING,
title: context.i18n.selectTrainerApp,
);
widget.onUpdate();
} else {
Navigator.push(
context,
MaterialPageRoute(
builder: (c) => ButtonSimulator(),
),
);
}
},
),
PrimaryButton(
leading: Icon(BCPage.customization.icon),
onPressed: () {
widget.goToNextPage();
},
child: Text(context.i18n.adjustControllerButtons),
),
],
),
],
],
),
),
);
}
}

View File

@@ -1,53 +0,0 @@
import 'package:flex_color_scheme/flex_color_scheme.dart';
import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart';
abstract final class AppTheme {
// The FlexColorScheme defined light mode ThemeData.
static ThemeData light = FlexThemeData.light(
// Using FlexColorScheme built-in FlexScheme enum based colors
scheme: FlexScheme.redM3,
primary: Color(0xFF0E74B7),
primaryContainer: Color(0x7C0E9297),
onPrimaryContainer: Colors.black,
// Component theme configurations for light mode.
subThemesData: const FlexSubThemesData(
interactionEffects: true,
tintedDisabledControls: true,
useM2StyleDividerInM3: true,
inputDecoratorIsFilled: true,
inputDecoratorBorderType: FlexInputBorderType.outline,
alignedDropdown: true,
navigationRailUseIndicator: true,
),
// Direct ThemeData properties.
visualDensity: FlexColorScheme.comfortablePlatformDensity,
cupertinoOverrideTheme: const CupertinoThemeData(applyThemeToAll: true),
);
// The FlexColorScheme defined dark mode ThemeData.
static ThemeData dark =
FlexThemeData.dark(
// Using FlexColorScheme built-in FlexScheme enum based colors.
scheme: FlexScheme.redM3,
// Component theme configurations for dark mode.
subThemesData: const FlexSubThemesData(
interactionEffects: true,
tintedDisabledControls: true,
blendOnColors: true,
useM2StyleDividerInM3: true,
inputDecoratorIsFilled: true,
inputDecoratorBorderType: FlexInputBorderType.outline,
alignedDropdown: true,
navigationRailUseIndicator: true,
),
// Direct ThemeData properties.
visualDensity: FlexColorScheme.comfortablePlatformDensity,
cupertinoOverrideTheme: const CupertinoThemeData(applyThemeToAll: true),
).copyWith(
scaffoldBackgroundColor: Color(0xff0b1623),
elevatedButtonTheme: ElevatedButtonThemeData(
style: ElevatedButton.styleFrom(backgroundColor: Colors.red, foregroundColor: Colors.white),
),
);
}

View File

@@ -1,13 +1,12 @@
import 'package:accessibility/accessibility.dart';
import 'package:bike_control/bluetooth/devices/hid/hid_device.dart';
import 'package:bike_control/utils/actions/base_actions.dart';
import 'package:bike_control/utils/core.dart';
import 'package:bike_control/utils/iap/iap_manager.dart';
import 'package:bike_control/utils/keymap/buttons.dart';
import 'package:bike_control/widgets/keymap_explanation.dart';
import 'package:dartx/dartx.dart';
import 'package:flutter/services.dart';
import 'package:swift_control/bluetooth/devices/hid/hid_device.dart';
import 'package:swift_control/bluetooth/devices/zwift/zwift_emulator.dart';
import 'package:swift_control/main.dart';
import 'package:swift_control/utils/actions/base_actions.dart';
import 'package:swift_control/utils/keymap/apps/custom_app.dart';
import 'package:swift_control/utils/keymap/buttons.dart';
import 'package:swift_control/widgets/keymap_explanation.dart';
import '../keymap/apps/supported_app.dart';
import '../single_line_exception.dart';
@@ -28,39 +27,33 @@ class AndroidActions extends BaseActions {
}
});
hidKeyPressed().listen((keyPressed) {
if (supportedApp is CustomApp) {
final button = supportedApp.keymap.getOrAddButton(keyPressed, () => ControllerButton(keyPressed));
hidKeyPressed().listen((keyPressed) async {
final hidDevice = HidDevice(keyPressed.source);
final button = hidDevice.getOrAddButton(keyPressed.hidKey, () => ControllerButton(keyPressed.hidKey));
final hidDevice = HidDevice('HID Device');
var availableDevice = connection.controllerDevices.firstOrNullWhere((e) => e.name == hidDevice.name);
if (availableDevice == null) {
connection.addDevices([hidDevice]);
availableDevice = hidDevice;
}
var availableDevice = core.connection.controllerDevices.firstOrNullWhere((e) => e.name == hidDevice.name);
if (availableDevice == null) {
core.connection.addDevices([hidDevice]);
availableDevice = hidDevice;
}
if (keyPressed.keyDown) {
availableDevice.handleButtonsClicked([button]);
} else if (keyPressed.keyUp) {
availableDevice.handleButtonsClicked([]);
}
});
}
@override
Future<String> performAction(ControllerButton button, {bool isKeyDown = true, bool isKeyUp = false}) async {
if (supportedApp == null) {
return ("Could not perform ${button.name.splitByUpperCase()}: No keymap set");
Future<ActionResult> performAction(ControllerButton button, {required bool isKeyDown, required bool isKeyUp}) async {
final superResult = await super.performAction(button, isKeyDown: isKeyDown, isKeyUp: isKeyUp);
if (superResult is! NotHandled) {
// Increment command count after successful execution
return superResult;
}
final keyPair = supportedApp!.keymap.getKeyPair(button)!;
final keyPair = supportedApp!.keymap.getKeyPair(button);
if (keyPair == null) {
return ("Could not perform ${button.name.splitByUpperCase()}: No action assigned");
}
if (keyPair.inGameAction != null && whooshLink.isConnected.value) {
return whooshLink.sendAction(keyPair.inGameAction!, keyPair.inGameActionValue);
} else if (keyPair.inGameAction != null && zwiftEmulator.isConnected.value) {
return zwiftEmulator.sendAction(keyPair.inGameAction!, keyPair.inGameActionValue);
} else if (keyPair.isSpecialKey) {
if (keyPair.isSpecialKey) {
await accessibilityHandler.controlMedia(switch (keyPair.physicalKey) {
PhysicalKeyboardKey.mediaTrackNext => MediaAction.next,
PhysicalKeyboardKey.mediaPlayPause => MediaAction.playPause,
@@ -68,7 +61,9 @@ class AndroidActions extends BaseActions {
PhysicalKeyboardKey.audioVolumeDown => MediaAction.volumeDown,
_ => throw SingleLineException("No action for key: ${keyPair.physicalKey}"),
});
return "Key pressed: ${keyPair.toString()}";
// Increment command count after successful execution
await IAPManager.instance.incrementCommandCount();
return Success("Key pressed: ${keyPair.toString()}");
}
final point = await resolveTouchPosition(keyPair: keyPair, windowInfo: windowInfo);
@@ -76,15 +71,19 @@ class AndroidActions extends BaseActions {
try {
await accessibilityHandler.performTouch(point.dx, point.dy, isKeyDown: isKeyDown, isKeyUp: isKeyUp);
} on PlatformException catch (e) {
return "Accessibility Service not working. Follow instructions at https://dontkillmyapp.com/";
return Error("Accessibility Service not working. Follow instructions at https://dontkillmyapp.com/");
}
return "Touch performed at: ${point.dx.toInt()}, ${point.dy.toInt()} -> ${isKeyDown && isKeyUp
? "click"
: isKeyDown
? "down"
: "up"}";
// Increment command count after successful execution
await IAPManager.instance.incrementCommandCount();
return Success(
"Touch performed at: ${point.dx.toInt()}, ${point.dy.toInt()} -> ${isKeyDown && isKeyUp
? "click"
: isKeyDown
? "down"
: "up"}",
);
}
return "No action assigned";
return NotHandled('No action assigned for ${button.toString().splitByUpperCase()}');
}
void ignoreHidDevices() {

Some files were not shown because too many files have changed in this diff Show More