6.0.0 (#74)
* 调整接口 * 临时提交 * 重构 Android 平台代码 * 临时提交 * 临时提交 * Android 6.0.0-dev.0 * 临时提交 * 实现 Windows 接口 * windows-6.0.0-dev.0 * Darwin 6.0.0-dev.0 * 临时提交 * 1 * 临时提交 * 调整接口 * windows-6.0.0-dev.1 * 临时提交 * interface-6.0.0-dev.7 * interface-6.0.0-dev.8 * 临时提交 * windows-6.0.0-dev.2 * 删除多余脚本 * interface-6.0.0-dev.9 * 临时提交 * 临时提交 * interface-6.0.0-dev.10 * android-6.0.0-dev.1 * windows-6.0.0-dev.3 * 临时提交 * interface-6.0.0-dev.11 * windows-6.0.0-dev.4 * 更新 pubspec.lock * 1 * interface-6.0.0-dev.12 * interface-6.0.0-dev.13 * interface-6.0.0-dev.14 * 临时提交 * interface-6.0.0-dev.15 * 临时提交 * interface-6.0.0-dev.16 * android-6.0.0-dev.2 * 临时提交 * windows-6.0.0-dev.5 * 临时提交 * 临时提交 * windows-6.0.0-dev.6 * 优化注释和代码样式 * 优化代码 * 临时提交 * 实现 Dart 接口 * darwin-6.0.0-dev.0 * linux-6.0.0-dev.0 * 修复已知问题 * 修复问题 * 6.0.0-dev.0 * 修改包名 * 更新版本 * 移除原生部分 * 临时提交 * 修复问题 * 更新 pigeon 19.0.0 * 更新 README,添加迁移文档 * linux-6.0.0-dev.1 * 解析扫描回复和扩展广播 * 修复 googletest 版本警告问题 * Use centralArgs instead of addressArgs * interface-6.0.0-dev.18 * android-6.0.0-dev.4 * linux-6.0.0-dev.2 * windows-6.0.0-dev.8 * darwin-6.0.0-dev.2 * 6.0.0-dev.1 * Update LICENSE * clang-format * Combine ADV_IND and SCAN_RES * TEMP commit: update exampe * Adjust advertisement combine logic * Implement `MyPeripheralMananger` on Windows * Added NuGet auto download and scan for names on peripheral (#67) * fetch nuget using other technique * move FetchContent to right location in CMakeLists.txt * also added hash for googletest --------- Co-authored-by: Kevin De Keyser <kevin@dekeyser.ch> * Fix errors. * Check BluetoothAdapter role supported state and implement PeripheralManager on Flutter side. * Sort code * Fix known errors * interface-6.0.0-dev.19 * windows-6.0.0-dev.9 * Optimize example * android-6.0.0-dev.5 * Optimize the Adverrtisement BottomSheet. * linux-6.0.0-dev.3 * Update dependency * Fix example errors. * Temp commit. * darwin-6.0.0-dev.3 * 6.0.0-dev.2 * Update README.md * 6.0.0 * darwin-6.0.0-dev.4 * android-6.0.0-dev.6 * 6.0.0-dev.3 * Update docs. * interface-6.0.0 * android-6.0.0 * darwin-6.0.0 * linux-6.0.0 * windows-6.0.0 * 6.0.0 * Update dependency --------- Co-authored-by: Kevin De Keyser <dekeyser.kevin97@gmail.com> Co-authored-by: Kevin De Keyser <kevin@dekeyser.ch>
This commit is contained in:
@ -27,7 +27,6 @@ migrate_working_dir/
|
||||
.dart_tool/
|
||||
.flutter-plugins
|
||||
.flutter-plugins-dependencies
|
||||
.packages
|
||||
.pub-cache/
|
||||
.pub/
|
||||
/build/
|
||||
|
@ -1,6 +1,6 @@
|
||||
# bluetooth_low_energy_example
|
||||
# bluetooth_low_energy_android_example
|
||||
|
||||
Demonstrates how to use the bluetooth_low_energy plugin.
|
||||
Demonstrates how to use the bluetooth_low_energy_android plugin.
|
||||
|
||||
## Getting Started
|
||||
|
||||
|
@ -1,68 +1,58 @@
|
||||
plugins {
|
||||
id "com.android.application"
|
||||
id "kotlin-android"
|
||||
// The Flutter Gradle Plugin must be applied after the Android and Kotlin Gradle plugins.
|
||||
id "dev.flutter.flutter-gradle-plugin"
|
||||
}
|
||||
|
||||
def localProperties = new Properties()
|
||||
def localPropertiesFile = rootProject.file('local.properties')
|
||||
def localPropertiesFile = rootProject.file("local.properties")
|
||||
if (localPropertiesFile.exists()) {
|
||||
localPropertiesFile.withReader('UTF-8') { reader ->
|
||||
localPropertiesFile.withReader("UTF-8") { reader ->
|
||||
localProperties.load(reader)
|
||||
}
|
||||
}
|
||||
|
||||
def flutterVersionCode = localProperties.getProperty('flutter.versionCode')
|
||||
def flutterVersionCode = localProperties.getProperty("flutter.versionCode")
|
||||
if (flutterVersionCode == null) {
|
||||
flutterVersionCode = '1'
|
||||
flutterVersionCode = "1"
|
||||
}
|
||||
|
||||
def flutterVersionName = localProperties.getProperty('flutter.versionName')
|
||||
def flutterVersionName = localProperties.getProperty("flutter.versionName")
|
||||
if (flutterVersionName == null) {
|
||||
flutterVersionName = '1.0'
|
||||
flutterVersionName = "1.0"
|
||||
}
|
||||
|
||||
android {
|
||||
namespace "dev.yanshouwang.bluetooth_low_energy_example"
|
||||
compileSdkVersion flutter.compileSdkVersion
|
||||
ndkVersion flutter.ndkVersion
|
||||
namespace = "dev.hebei.bluetooth_low_energy_android_example"
|
||||
compileSdk = flutter.compileSdkVersion
|
||||
ndkVersion = flutter.ndkVersion
|
||||
|
||||
compileOptions {
|
||||
sourceCompatibility JavaVersion.VERSION_1_8
|
||||
targetCompatibility JavaVersion.VERSION_1_8
|
||||
}
|
||||
|
||||
kotlinOptions {
|
||||
jvmTarget = '1.8'
|
||||
}
|
||||
|
||||
sourceSets {
|
||||
main.java.srcDirs += 'src/main/kotlin'
|
||||
sourceCompatibility = JavaVersion.VERSION_1_8
|
||||
targetCompatibility = JavaVersion.VERSION_1_8
|
||||
}
|
||||
|
||||
defaultConfig {
|
||||
// TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html).
|
||||
applicationId "dev.yanshouwang.bluetooth_low_energy_example"
|
||||
applicationId = "dev.hebei.bluetooth_low_energy_android_example"
|
||||
// You can update the following values to match your application needs.
|
||||
// For more information, see: https://docs.flutter.dev/deployment/android#reviewing-the-gradle-build-configuration.
|
||||
// minSdkVersion flutter.minSdkVersion
|
||||
minSdkVersion 21
|
||||
targetSdkVersion flutter.targetSdkVersion
|
||||
versionCode flutterVersionCode.toInteger()
|
||||
versionName flutterVersionName
|
||||
minSdk = flutter.minSdkVersion
|
||||
targetSdk = flutter.targetSdkVersion
|
||||
versionCode = flutterVersionCode.toInteger()
|
||||
versionName = flutterVersionName
|
||||
}
|
||||
|
||||
buildTypes {
|
||||
release {
|
||||
// TODO: Add your own signing config for the release build.
|
||||
// Signing with the debug keys for now, so `flutter run --release` works.
|
||||
signingConfig signingConfigs.debug
|
||||
signingConfig = signingConfigs.debug
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
flutter {
|
||||
source '../..'
|
||||
source = "../.."
|
||||
}
|
||||
|
||||
dependencies {}
|
||||
|
@ -1,12 +1,13 @@
|
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<application
|
||||
android:label="bluetooth_low_energy_example"
|
||||
android:label="bluetooth_low_energy_android_example"
|
||||
android:name="${applicationName}"
|
||||
android:icon="@mipmap/ic_launcher">
|
||||
<activity
|
||||
android:name=".MainActivity"
|
||||
android:exported="true"
|
||||
android:launchMode="singleTop"
|
||||
android:taskAffinity=""
|
||||
android:theme="@style/LaunchTheme"
|
||||
android:configChanges="orientation|keyboardHidden|keyboard|screenSize|smallestScreenSize|locale|layoutDirection|fontScale|screenLayout|density|uiMode"
|
||||
android:hardwareAccelerated="true"
|
||||
@ -30,4 +31,15 @@
|
||||
android:name="flutterEmbedding"
|
||||
android:value="2" />
|
||||
</application>
|
||||
<!-- Required to query activities that can process text, see:
|
||||
https://developer.android.com/training/package-visibility and
|
||||
https://developer.android.com/reference/android/content/Intent#ACTION_PROCESS_TEXT.
|
||||
|
||||
In particular, this is used by the Flutter engine in io.flutter.plugin.text.ProcessTextPlugin. -->
|
||||
<queries>
|
||||
<intent>
|
||||
<action android:name="android.intent.action.PROCESS_TEXT"/>
|
||||
<data android:mimeType="text/plain"/>
|
||||
</intent>
|
||||
</queries>
|
||||
</manifest>
|
||||
|
@ -0,0 +1,5 @@
|
||||
package dev.hebei.bluetooth_low_energy_android_example
|
||||
|
||||
import io.flutter.embedding.android.FlutterActivity
|
||||
|
||||
class MainActivity: FlutterActivity()
|
@ -1,6 +0,0 @@
|
||||
package dev.yanshouwang.bluetooth_low_energy_example
|
||||
|
||||
import io.flutter.embedding.android.FlutterActivity
|
||||
|
||||
class MainActivity: FlutterActivity() {
|
||||
}
|
@ -1,16 +1,3 @@
|
||||
buildscript {
|
||||
ext.kotlin_version = '1.7.10'
|
||||
repositories {
|
||||
google()
|
||||
mavenCentral()
|
||||
}
|
||||
|
||||
dependencies {
|
||||
classpath 'com.android.tools.build:gradle:7.3.0'
|
||||
classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
|
||||
}
|
||||
}
|
||||
|
||||
allprojects {
|
||||
repositories {
|
||||
google()
|
||||
@ -18,12 +5,12 @@ allprojects {
|
||||
}
|
||||
}
|
||||
|
||||
rootProject.buildDir = '../build'
|
||||
rootProject.buildDir = "../build"
|
||||
subprojects {
|
||||
project.buildDir = "${rootProject.buildDir}/${project.name}"
|
||||
}
|
||||
subprojects {
|
||||
project.evaluationDependsOn(':app')
|
||||
project.evaluationDependsOn(":app")
|
||||
}
|
||||
|
||||
tasks.register("clean", Delete) {
|
||||
|
@ -1,3 +1,3 @@
|
||||
org.gradle.jvmargs=-Xmx1536M
|
||||
org.gradle.jvmargs=-Xmx4G -XX:+HeapDumpOnOutOfMemoryError
|
||||
android.useAndroidX=true
|
||||
android.enableJetifier=true
|
||||
|
@ -2,4 +2,4 @@ distributionBase=GRADLE_USER_HOME
|
||||
distributionPath=wrapper/dists
|
||||
zipStoreBase=GRADLE_USER_HOME
|
||||
zipStorePath=wrapper/dists
|
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-7.5-all.zip
|
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-7.6.3-all.zip
|
||||
|
@ -5,16 +5,21 @@ pluginManagement {
|
||||
def flutterSdkPath = properties.getProperty("flutter.sdk")
|
||||
assert flutterSdkPath != null, "flutter.sdk not set in local.properties"
|
||||
return flutterSdkPath
|
||||
}
|
||||
settings.ext.flutterSdkPath = flutterSdkPath()
|
||||
}()
|
||||
|
||||
includeBuild("${settings.ext.flutterSdkPath}/packages/flutter_tools/gradle")
|
||||
includeBuild("$flutterSdkPath/packages/flutter_tools/gradle")
|
||||
|
||||
plugins {
|
||||
id "dev.flutter.flutter-gradle-plugin" version "1.0.0" apply false
|
||||
repositories {
|
||||
google()
|
||||
mavenCentral()
|
||||
gradlePluginPortal()
|
||||
}
|
||||
}
|
||||
|
||||
include ":app"
|
||||
plugins {
|
||||
id "dev.flutter.flutter-plugin-loader" version "1.0.0"
|
||||
id "com.android.application" version "7.3.0" apply false
|
||||
id "org.jetbrains.kotlin.android" version "1.7.10" apply false
|
||||
}
|
||||
|
||||
apply from: "${settings.ext.flutterSdkPath}/packages/flutter_tools/gradle/app_plugin_loader.gradle"
|
||||
include ":app"
|
||||
|
@ -6,17 +6,8 @@
|
||||
// For more information about Flutter integration tests, please see
|
||||
// https://docs.flutter.dev/cookbook/testing/integration/introduction
|
||||
|
||||
// import 'package:flutter_test/flutter_test.dart';
|
||||
import 'package:integration_test/integration_test.dart';
|
||||
|
||||
void main() {
|
||||
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
|
||||
|
||||
// testWidgets('getPlatformVersion test', (WidgetTester tester) async {
|
||||
// final BluetoothLowEnergy plugin = BluetoothLowEnergy();
|
||||
// final String? version = await plugin.getPlatformVersion();
|
||||
// // The version string depends on the host platform running the test, so
|
||||
// // just assert that some non-empty string is returned.
|
||||
// expect(version?.isNotEmpty, true);
|
||||
// });
|
||||
}
|
||||
|
File diff suppressed because it is too large
Load Diff
1
bluetooth_low_energy_android/example/lib/models.dart
Normal file
1
bluetooth_low_energy_android/example/lib/models.dart
Normal file
@ -0,0 +1 @@
|
||||
export 'models/log.dart';
|
10
bluetooth_low_energy_android/example/lib/models/log.dart
Normal file
10
bluetooth_low_energy_android/example/lib/models/log.dart
Normal file
@ -0,0 +1,10 @@
|
||||
class Log {
|
||||
final DateTime time;
|
||||
final String type;
|
||||
final String message;
|
||||
|
||||
Log({
|
||||
required this.type,
|
||||
required this.message,
|
||||
}) : time = DateTime.now();
|
||||
}
|
85
bluetooth_low_energy_android/example/lib/router_config.dart
Normal file
85
bluetooth_low_energy_android/example/lib/router_config.dart
Normal file
@ -0,0 +1,85 @@
|
||||
import 'package:bluetooth_low_energy_platform_interface/bluetooth_low_energy_platform_interface.dart';
|
||||
import 'package:clover/clover.dart';
|
||||
import 'package:collection/collection.dart';
|
||||
import 'package:go_router/go_router.dart';
|
||||
|
||||
import 'view_models.dart';
|
||||
import 'views.dart';
|
||||
|
||||
final routerConfig = GoRouter(
|
||||
redirect: (context, state) {
|
||||
if (state.matchedLocation == '/') {
|
||||
return '/central';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
routes: [
|
||||
StatefulShellRoute(
|
||||
// builder: (context, state, navigationShell) {
|
||||
// return HomeView(navigationShell: navigationShell);
|
||||
// },
|
||||
builder: (context, state, navigationShell) => navigationShell,
|
||||
navigatorContainerBuilder: (context, navigationShell, children) {
|
||||
final navigators = children.mapIndexed(
|
||||
(index, element) {
|
||||
if (index == 0) {
|
||||
return ViewModelBinding(
|
||||
viewBuilder: (context) => element,
|
||||
viewModelBuilder: (context) => CentralManagerViewModel(),
|
||||
);
|
||||
} else {
|
||||
return element;
|
||||
}
|
||||
},
|
||||
).toList();
|
||||
return HomeView(
|
||||
navigationShell: navigationShell,
|
||||
navigators: navigators,
|
||||
);
|
||||
},
|
||||
branches: [
|
||||
StatefulShellBranch(
|
||||
routes: [
|
||||
GoRoute(
|
||||
path: '/central',
|
||||
builder: (context, state) {
|
||||
return const CentralManagerView();
|
||||
},
|
||||
routes: [
|
||||
GoRoute(
|
||||
path: ':uuid',
|
||||
builder: (context, state) {
|
||||
final uuidValue = state.pathParameters['uuid']!;
|
||||
final uuid = UUID.fromString(uuidValue);
|
||||
final viewModel =
|
||||
ViewModel.of<CentralManagerViewModel>(context);
|
||||
final eventArgs = viewModel.discoveries.firstWhere(
|
||||
(discovery) => discovery.peripheral.uuid == uuid);
|
||||
return ViewModelBinding(
|
||||
viewBuilder: (context) => PeripheralView(),
|
||||
viewModelBuilder: (context) =>
|
||||
PeripheralViewModel(eventArgs),
|
||||
);
|
||||
},
|
||||
),
|
||||
],
|
||||
),
|
||||
],
|
||||
),
|
||||
StatefulShellBranch(
|
||||
routes: [
|
||||
GoRoute(
|
||||
path: '/peripheral',
|
||||
builder: (context, state) {
|
||||
return ViewModelBinding(
|
||||
viewBuilder: (context) => const PeripheralManagerView(),
|
||||
viewModelBuilder: (context) => PeripheralManagerViewModel(),
|
||||
);
|
||||
},
|
||||
),
|
||||
],
|
||||
),
|
||||
],
|
||||
),
|
||||
],
|
||||
);
|
@ -0,0 +1,6 @@
|
||||
export 'view_models/central_manager_view_model.dart';
|
||||
export 'view_models/peripheral_view_model.dart';
|
||||
export 'view_models/service_view_model.dart';
|
||||
export 'view_models/characteristic_view_model.dart';
|
||||
export 'view_models/descriptor_view_model.dart';
|
||||
export 'view_models/peripheral_manager_view_model.dart';
|
@ -0,0 +1,74 @@
|
||||
import 'dart:async';
|
||||
|
||||
import 'package:bluetooth_low_energy_platform_interface/bluetooth_low_energy_platform_interface.dart';
|
||||
import 'package:clover/clover.dart';
|
||||
import 'package:logging/logging.dart';
|
||||
|
||||
class CentralManagerViewModel extends ViewModel {
|
||||
final CentralManager _manager;
|
||||
final List<DiscoveredEventArgs> _discoveries;
|
||||
bool _discovering;
|
||||
|
||||
late final StreamSubscription _stateChangedSubscription;
|
||||
late final StreamSubscription _discoveredSubscription;
|
||||
|
||||
CentralManagerViewModel()
|
||||
: _manager = CentralManager()..logLevel = Level.INFO,
|
||||
_discoveries = [],
|
||||
_discovering = false {
|
||||
_stateChangedSubscription = _manager.stateChanged.listen((eventArgs) async {
|
||||
if (eventArgs.state == BluetoothLowEnergyState.unauthorized) {
|
||||
await _manager.authorize();
|
||||
}
|
||||
notifyListeners();
|
||||
});
|
||||
_discoveredSubscription = _manager.discovered.listen((eventArgs) {
|
||||
final peripheral = eventArgs.peripheral;
|
||||
final index = _discoveries.indexWhere((i) => i.peripheral == peripheral);
|
||||
if (index < 0) {
|
||||
_discoveries.add(eventArgs);
|
||||
} else {
|
||||
_discoveries[index] = eventArgs;
|
||||
}
|
||||
notifyListeners();
|
||||
});
|
||||
}
|
||||
|
||||
BluetoothLowEnergyState get state => _manager.state;
|
||||
bool get discovering => _discovering;
|
||||
List<DiscoveredEventArgs> get discoveries => _discoveries;
|
||||
|
||||
Future<void> showAppSettings() async {
|
||||
await _manager.showAppSettings();
|
||||
}
|
||||
|
||||
Future<void> startDiscovery({
|
||||
List<UUID>? serviceUUIDs,
|
||||
}) async {
|
||||
if (_discovering) {
|
||||
return;
|
||||
}
|
||||
_discoveries.clear();
|
||||
await _manager.startDiscovery(
|
||||
serviceUUIDs: serviceUUIDs,
|
||||
);
|
||||
_discovering = true;
|
||||
notifyListeners();
|
||||
}
|
||||
|
||||
Future<void> stopDiscovery() async {
|
||||
if (!_discovering) {
|
||||
return;
|
||||
}
|
||||
await _manager.stopDiscovery();
|
||||
_discovering = false;
|
||||
notifyListeners();
|
||||
}
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
_stateChangedSubscription.cancel();
|
||||
_discoveredSubscription.cancel();
|
||||
super.dispose();
|
||||
}
|
||||
}
|
@ -0,0 +1,145 @@
|
||||
import 'dart:async';
|
||||
import 'dart:typed_data';
|
||||
|
||||
import 'package:bluetooth_low_energy_platform_interface/bluetooth_low_energy_platform_interface.dart';
|
||||
import 'package:bluetooth_low_energy_android_example/models.dart';
|
||||
import 'package:clover/clover.dart';
|
||||
|
||||
import 'descriptor_view_model.dart';
|
||||
|
||||
class CharacteristicViewModel extends ViewModel {
|
||||
final CentralManager _manager;
|
||||
final Peripheral _peripheral;
|
||||
final GATTCharacteristic _characteristic;
|
||||
final List<DescriptorViewModel> _descriptorViewModels;
|
||||
final List<Log> _logs;
|
||||
|
||||
GATTCharacteristicWriteType _writeType;
|
||||
bool _notifyState;
|
||||
|
||||
late final StreamSubscription _characteristicNotifiedSubscription;
|
||||
|
||||
CharacteristicViewModel({
|
||||
required CentralManager manager,
|
||||
required Peripheral peripheral,
|
||||
required GATTCharacteristic characteristic,
|
||||
}) : _manager = manager,
|
||||
_peripheral = peripheral,
|
||||
_characteristic = characteristic,
|
||||
_descriptorViewModels = characteristic.descriptors
|
||||
.map((descriptor) => DescriptorViewModel(descriptor))
|
||||
.toList(),
|
||||
_logs = [],
|
||||
_writeType = GATTCharacteristicWriteType.withResponse,
|
||||
_notifyState = false {
|
||||
if (!canWrite && canWriteWithoutResponse) {
|
||||
_writeType = GATTCharacteristicWriteType.withoutResponse;
|
||||
}
|
||||
_characteristicNotifiedSubscription =
|
||||
_manager.characteristicNotified.listen((eventArgs) {
|
||||
if (eventArgs.characteristic != _characteristic) {
|
||||
return;
|
||||
}
|
||||
final log = Log(
|
||||
type: 'Notified',
|
||||
message: '[${eventArgs.value.length}] ${eventArgs.value}',
|
||||
);
|
||||
_logs.add(log);
|
||||
notifyListeners();
|
||||
});
|
||||
}
|
||||
|
||||
UUID get uuid => _characteristic.uuid;
|
||||
bool get canRead =>
|
||||
_characteristic.properties.contains(GATTCharacteristicProperty.read);
|
||||
bool get canWrite =>
|
||||
_characteristic.properties.contains(GATTCharacteristicProperty.write);
|
||||
bool get canWriteWithoutResponse => _characteristic.properties
|
||||
.contains(GATTCharacteristicProperty.writeWithoutResponse);
|
||||
bool get canNotify =>
|
||||
_characteristic.properties.contains(GATTCharacteristicProperty.notify) ||
|
||||
_characteristic.properties.contains(GATTCharacteristicProperty.indicate);
|
||||
List<DescriptorViewModel> get descriptorViewModels => _descriptorViewModels;
|
||||
List<Log> get logs => _logs;
|
||||
GATTCharacteristicWriteType get writeType => _writeType;
|
||||
bool get notifyState => _notifyState;
|
||||
|
||||
Future<void> read() async {
|
||||
final value = await _manager.readCharacteristic(
|
||||
_peripheral,
|
||||
_characteristic,
|
||||
);
|
||||
final log = Log(
|
||||
type: 'Read',
|
||||
message: '[${value.length}] $value',
|
||||
);
|
||||
_logs.add(log);
|
||||
notifyListeners();
|
||||
}
|
||||
|
||||
void setWriteType(GATTCharacteristicWriteType type) {
|
||||
if (type == GATTCharacteristicWriteType.withResponse && !canWrite) {
|
||||
throw ArgumentError.value(type);
|
||||
}
|
||||
if (type == GATTCharacteristicWriteType.withoutResponse &&
|
||||
!canWriteWithoutResponse) {
|
||||
throw ArgumentError.value(type);
|
||||
}
|
||||
_writeType = type;
|
||||
notifyListeners();
|
||||
}
|
||||
|
||||
Future<void> write(Uint8List value) async {
|
||||
// Fragments the value by maximumWriteLength.
|
||||
final fragmentSize = await _manager.getMaximumWriteLength(
|
||||
_peripheral,
|
||||
type: writeType,
|
||||
);
|
||||
var start = 0;
|
||||
while (start < value.length) {
|
||||
final end = start + fragmentSize;
|
||||
final fragmentedValue =
|
||||
end < value.length ? value.sublist(start, end) : value.sublist(start);
|
||||
final type = writeType;
|
||||
await _manager.writeCharacteristic(
|
||||
_peripheral,
|
||||
_characteristic,
|
||||
value: fragmentedValue,
|
||||
type: type,
|
||||
);
|
||||
final log = Log(
|
||||
type: type == GATTCharacteristicWriteType.withResponse
|
||||
? 'Write'
|
||||
: 'Write without response',
|
||||
message: '[${value.length}] $value',
|
||||
);
|
||||
_logs.add(log);
|
||||
notifyListeners();
|
||||
start = end;
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> setNotifyState(bool state) async {
|
||||
await _manager.setCharacteristicNotifyState(
|
||||
_peripheral,
|
||||
_characteristic,
|
||||
state: state,
|
||||
);
|
||||
_notifyState = state;
|
||||
notifyListeners();
|
||||
}
|
||||
|
||||
void clearLogs() {
|
||||
_logs.clear();
|
||||
notifyListeners();
|
||||
}
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
_characteristicNotifiedSubscription.cancel();
|
||||
for (var descriptorViewModel in descriptorViewModels) {
|
||||
descriptorViewModel.dispose();
|
||||
}
|
||||
super.dispose();
|
||||
}
|
||||
}
|
@ -0,0 +1,10 @@
|
||||
import 'package:bluetooth_low_energy_platform_interface/bluetooth_low_energy_platform_interface.dart';
|
||||
import 'package:clover/clover.dart';
|
||||
|
||||
class DescriptorViewModel extends ViewModel {
|
||||
final GATTDescriptor _descriptor;
|
||||
|
||||
DescriptorViewModel(this._descriptor);
|
||||
|
||||
UUID get uuid => _descriptor.uuid;
|
||||
}
|
@ -0,0 +1,170 @@
|
||||
import 'dart:async';
|
||||
import 'dart:typed_data';
|
||||
|
||||
import 'package:bluetooth_low_energy_platform_interface/bluetooth_low_energy_platform_interface.dart';
|
||||
import 'package:bluetooth_low_energy_android_example/models.dart';
|
||||
import 'package:clover/clover.dart';
|
||||
import 'package:logging/logging.dart';
|
||||
|
||||
class PeripheralManagerViewModel extends ViewModel {
|
||||
final PeripheralManager _manager;
|
||||
final List<Log> _logs;
|
||||
bool _advertising;
|
||||
|
||||
late final StreamSubscription _stateChangedSubscription;
|
||||
late final StreamSubscription _characteristicReadRequestedSubscription;
|
||||
late final StreamSubscription _characteristicWriteRequestedSubscription;
|
||||
late final StreamSubscription _characteristicNotifyStateChangedSubscription;
|
||||
|
||||
PeripheralManagerViewModel()
|
||||
: _manager = PeripheralManager()..logLevel = Level.INFO,
|
||||
_logs = [],
|
||||
_advertising = false {
|
||||
_stateChangedSubscription = _manager.stateChanged.listen((eventArgs) async {
|
||||
if (eventArgs.state == BluetoothLowEnergyState.unauthorized) {
|
||||
await _manager.authorize();
|
||||
}
|
||||
notifyListeners();
|
||||
});
|
||||
_characteristicReadRequestedSubscription =
|
||||
_manager.characteristicReadRequested.listen((eventArgs) async {
|
||||
final central = eventArgs.central;
|
||||
final characteristic = eventArgs.characteristic;
|
||||
final request = eventArgs.request;
|
||||
final offset = request.offset;
|
||||
final log = Log(
|
||||
type: 'Characteristic read requested',
|
||||
message: '${central.uuid}, ${characteristic.uuid}, $offset',
|
||||
);
|
||||
_logs.add(log);
|
||||
notifyListeners();
|
||||
final elements = List.generate(100, (i) => i % 256);
|
||||
final value = Uint8List.fromList(elements);
|
||||
final trimmedValue = value.sublist(offset);
|
||||
await _manager.respondReadRequestWithValue(
|
||||
request,
|
||||
value: trimmedValue,
|
||||
);
|
||||
});
|
||||
_characteristicWriteRequestedSubscription =
|
||||
_manager.characteristicWriteRequested.listen((eventArgs) async {
|
||||
final central = eventArgs.central;
|
||||
final characteristic = eventArgs.characteristic;
|
||||
final request = eventArgs.request;
|
||||
final offset = request.offset;
|
||||
final value = request.value;
|
||||
final log = Log(
|
||||
type: 'Characteristic write requested',
|
||||
message:
|
||||
'[${value.length}] ${central.uuid}, ${characteristic.uuid}, $offset, $value',
|
||||
);
|
||||
_logs.add(log);
|
||||
notifyListeners();
|
||||
await _manager.respondWriteRequest(request);
|
||||
});
|
||||
_characteristicNotifyStateChangedSubscription =
|
||||
_manager.characteristicNotifyStateChanged.listen((eventArgs) async {
|
||||
final central = eventArgs.central;
|
||||
final characteristic = eventArgs.characteristic;
|
||||
final state = eventArgs.state;
|
||||
final log = Log(
|
||||
type: 'Characteristic notify state changed',
|
||||
message: '${central.uuid}, ${characteristic.uuid}, $state',
|
||||
);
|
||||
_logs.add(log);
|
||||
notifyListeners();
|
||||
// Write someting to the central when notify started.
|
||||
if (state) {
|
||||
final maximumNotifyLength =
|
||||
await _manager.getMaximumNotifyLength(central);
|
||||
final elements = List.generate(maximumNotifyLength, (i) => i % 256);
|
||||
final value = Uint8List.fromList(elements);
|
||||
await _manager.notifyCharacteristic(
|
||||
central,
|
||||
characteristic,
|
||||
value: value,
|
||||
);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
BluetoothLowEnergyState get state => _manager.state;
|
||||
bool get advertising => _advertising;
|
||||
List<Log> get logs => _logs;
|
||||
|
||||
Future<void> showAppSettings() async {
|
||||
await _manager.showAppSettings();
|
||||
}
|
||||
|
||||
Future<void> startAdvertising() async {
|
||||
if (_advertising) {
|
||||
return;
|
||||
}
|
||||
await _manager.removeAllServices();
|
||||
final elements = List.generate(100, (i) => i % 256);
|
||||
final value = Uint8List.fromList(elements);
|
||||
final service = GATTService(
|
||||
uuid: UUID.short(100),
|
||||
isPrimary: true,
|
||||
includedServices: [],
|
||||
characteristics: [
|
||||
GATTCharacteristic.immutable(
|
||||
uuid: UUID.short(200),
|
||||
value: value,
|
||||
descriptors: [],
|
||||
),
|
||||
GATTCharacteristic.mutable(
|
||||
uuid: UUID.short(201),
|
||||
properties: [
|
||||
GATTCharacteristicProperty.read,
|
||||
GATTCharacteristicProperty.write,
|
||||
GATTCharacteristicProperty.writeWithoutResponse,
|
||||
GATTCharacteristicProperty.notify,
|
||||
GATTCharacteristicProperty.indicate,
|
||||
],
|
||||
permissions: [
|
||||
GATTCharacteristicPermission.read,
|
||||
GATTCharacteristicPermission.write,
|
||||
],
|
||||
descriptors: [],
|
||||
),
|
||||
],
|
||||
);
|
||||
await _manager.addService(service);
|
||||
final advertisement = Advertisement(
|
||||
name: 'BLE-12138',
|
||||
manufacturerSpecificData: [
|
||||
ManufacturerSpecificData(
|
||||
id: 0x2e19,
|
||||
data: Uint8List.fromList([0x01, 0x02, 0x03]),
|
||||
)
|
||||
],
|
||||
);
|
||||
await _manager.startAdvertising(advertisement);
|
||||
_advertising = true;
|
||||
notifyListeners();
|
||||
}
|
||||
|
||||
Future<void> stopAdvertising() async {
|
||||
if (!_advertising) {
|
||||
return;
|
||||
}
|
||||
await _manager.stopAdvertising();
|
||||
_advertising = false;
|
||||
notifyListeners();
|
||||
}
|
||||
|
||||
void clearLogs() {
|
||||
_logs.clear();
|
||||
notifyListeners();
|
||||
}
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
_stateChangedSubscription.cancel();
|
||||
_characteristicReadRequestedSubscription.cancel();
|
||||
_characteristicWriteRequestedSubscription.cancel();
|
||||
_characteristicNotifyStateChangedSubscription.cancel();
|
||||
super.dispose();
|
||||
}
|
||||
}
|
@ -0,0 +1,83 @@
|
||||
import 'dart:async';
|
||||
|
||||
import 'package:bluetooth_low_energy_platform_interface/bluetooth_low_energy_platform_interface.dart';
|
||||
import 'package:clover/clover.dart';
|
||||
import 'package:hybrid_logging/hybrid_logging.dart';
|
||||
|
||||
import 'service_view_model.dart';
|
||||
|
||||
class PeripheralViewModel extends ViewModel with TypeLogger {
|
||||
final CentralManager _manager;
|
||||
final Peripheral _peripheral;
|
||||
final String? _name;
|
||||
bool _connected;
|
||||
List<ServiceViewModel> _serviceViewModels;
|
||||
|
||||
late final StreamSubscription _connectionStateChangedSubscription;
|
||||
late final StreamSubscription _mtuChangedChangedSubscription;
|
||||
|
||||
PeripheralViewModel(DiscoveredEventArgs eventArgs)
|
||||
: _manager = CentralManager(),
|
||||
_peripheral = eventArgs.peripheral,
|
||||
_name = eventArgs.advertisement.name,
|
||||
_connected = false,
|
||||
_serviceViewModels = [] {
|
||||
_connectionStateChangedSubscription =
|
||||
_manager.connectionStateChanged.listen((eventArgs) {
|
||||
if (eventArgs.peripheral != _peripheral) {
|
||||
return;
|
||||
}
|
||||
if (eventArgs.state == ConnectionState.connected) {
|
||||
_connected = true;
|
||||
} else {
|
||||
_connected = false;
|
||||
_serviceViewModels = [];
|
||||
}
|
||||
notifyListeners();
|
||||
});
|
||||
_mtuChangedChangedSubscription = _manager.mtuChanged.listen((eventArgs) {
|
||||
if (eventArgs.peripheral != _peripheral) {
|
||||
return;
|
||||
}
|
||||
logger.info('MTU chanaged: ${eventArgs.mtu}');
|
||||
});
|
||||
}
|
||||
|
||||
UUID get uuid => _peripheral.uuid;
|
||||
String? get name => _name;
|
||||
bool get connected => _connected;
|
||||
List<ServiceViewModel> get serviceViewModels => _serviceViewModels;
|
||||
|
||||
Future<void> connect() async {
|
||||
await _manager.connect(_peripheral);
|
||||
}
|
||||
|
||||
Future<void> disconnect() async {
|
||||
await _manager.disconnect(_peripheral);
|
||||
}
|
||||
|
||||
Future<void> discoverGATT() async {
|
||||
final services = await _manager.discoverGATT(_peripheral);
|
||||
_serviceViewModels = services
|
||||
.map((service) => ServiceViewModel(
|
||||
manager: _manager,
|
||||
peripheral: _peripheral,
|
||||
service: service,
|
||||
))
|
||||
.toList();
|
||||
notifyListeners();
|
||||
}
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
if (connected) {
|
||||
disconnect();
|
||||
}
|
||||
_connectionStateChangedSubscription.cancel();
|
||||
_mtuChangedChangedSubscription.cancel();
|
||||
for (var serviceViewModel in serviceViewModels) {
|
||||
serviceViewModel.dispose();
|
||||
}
|
||||
super.dispose();
|
||||
}
|
||||
}
|
@ -0,0 +1,46 @@
|
||||
import 'package:bluetooth_low_energy_platform_interface/bluetooth_low_energy_platform_interface.dart';
|
||||
import 'package:clover/clover.dart';
|
||||
|
||||
import 'characteristic_view_model.dart';
|
||||
|
||||
class ServiceViewModel extends ViewModel {
|
||||
final GATTService _service;
|
||||
|
||||
final List<ServiceViewModel> _includedServiceViewModels;
|
||||
final List<CharacteristicViewModel> _characteristicViewModels;
|
||||
|
||||
ServiceViewModel({
|
||||
required CentralManager manager,
|
||||
required Peripheral peripheral,
|
||||
required GATTService service,
|
||||
}) : _service = service,
|
||||
_includedServiceViewModels = service.includedServices
|
||||
.map((service) => ServiceViewModel(
|
||||
manager: manager,
|
||||
peripheral: peripheral,
|
||||
service: service,
|
||||
))
|
||||
.toList(),
|
||||
_characteristicViewModels = service.characteristics
|
||||
.map((characteristic) => CharacteristicViewModel(
|
||||
manager: manager,
|
||||
peripheral: peripheral,
|
||||
characteristic: characteristic,
|
||||
))
|
||||
.toList();
|
||||
|
||||
UUID get uuid => _service.uuid;
|
||||
bool get isPrimary => _service.isPrimary;
|
||||
List<ServiceViewModel> get includedServiceViewModels =>
|
||||
_includedServiceViewModels;
|
||||
List<CharacteristicViewModel> get characteristicViewModels =>
|
||||
_characteristicViewModels;
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
for (var characteristicViewModel in characteristicViewModels) {
|
||||
characteristicViewModel.dispose();
|
||||
}
|
||||
super.dispose();
|
||||
}
|
||||
}
|
4
bluetooth_low_energy_android/example/lib/views.dart
Normal file
4
bluetooth_low_energy_android/example/lib/views.dart
Normal file
@ -0,0 +1,4 @@
|
||||
export 'views/home_view.dart';
|
||||
export 'views/central_manager_view.dart';
|
||||
export 'views/peripheral_manager_view.dart';
|
||||
export 'views/peripheral_view.dart';
|
@ -0,0 +1,69 @@
|
||||
import 'package:bluetooth_low_energy_platform_interface/bluetooth_low_energy_platform_interface.dart';
|
||||
import 'package:convert/convert.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
class AdvertisementView extends StatelessWidget {
|
||||
final Advertisement advertisement;
|
||||
|
||||
const AdvertisementView({
|
||||
super.key,
|
||||
required this.advertisement,
|
||||
});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final manufacturerSpecificData = advertisement.manufacturerSpecificData;
|
||||
return LayoutBuilder(
|
||||
builder: (context, constraints) {
|
||||
const idWidth = 100.0;
|
||||
final valueWidth = constraints.maxWidth - idWidth - 16.0 * 2.0;
|
||||
return DataTable(
|
||||
columnSpacing: 0.0,
|
||||
horizontalMargin: 16.0,
|
||||
columns: [
|
||||
DataColumn(
|
||||
label: Container(
|
||||
width: idWidth,
|
||||
alignment: Alignment.center,
|
||||
child: const Text('Id'),
|
||||
),
|
||||
),
|
||||
DataColumn(
|
||||
label: Container(
|
||||
width: valueWidth,
|
||||
alignment: Alignment.center,
|
||||
child: const Text('Value'),
|
||||
),
|
||||
),
|
||||
],
|
||||
rows: manufacturerSpecificData.map((item) {
|
||||
final id = '0x${item.id.toRadixString(16).padLeft(4, '0')}';
|
||||
final value = hex.encode(item.data);
|
||||
return DataRow(
|
||||
cells: [
|
||||
DataCell(
|
||||
Container(
|
||||
width: idWidth,
|
||||
alignment: Alignment.center,
|
||||
child: Text(id),
|
||||
),
|
||||
),
|
||||
DataCell(
|
||||
Container(
|
||||
width: valueWidth,
|
||||
alignment: Alignment.center,
|
||||
child: Text(
|
||||
value,
|
||||
maxLines: 2,
|
||||
overflow: TextOverflow.ellipsis,
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
);
|
||||
}).toList(),
|
||||
);
|
||||
},
|
||||
);
|
||||
}
|
||||
}
|
@ -0,0 +1,124 @@
|
||||
import 'package:bluetooth_low_energy_platform_interface/bluetooth_low_energy_platform_interface.dart';
|
||||
import 'package:bluetooth_low_energy_android_example/view_models.dart';
|
||||
import 'package:bluetooth_low_energy_android_example/widgets.dart';
|
||||
import 'package:clover/clover.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:go_router/go_router.dart';
|
||||
|
||||
import 'advertisement_view.dart';
|
||||
|
||||
class CentralManagerView extends StatelessWidget {
|
||||
const CentralManagerView({super.key});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final viewModel = ViewModel.of<CentralManagerViewModel>(context);
|
||||
final state = viewModel.state;
|
||||
final discovering = viewModel.discovering;
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: const Text('Central Manager'),
|
||||
actions: [
|
||||
TextButton(
|
||||
onPressed: state == BluetoothLowEnergyState.poweredOn
|
||||
? () async {
|
||||
if (discovering) {
|
||||
await viewModel.stopDiscovery();
|
||||
} else {
|
||||
await viewModel.startDiscovery();
|
||||
}
|
||||
}
|
||||
: null,
|
||||
child: Text(discovering ? 'END' : 'BEGIN'),
|
||||
),
|
||||
],
|
||||
),
|
||||
body: buildBody(context),
|
||||
);
|
||||
}
|
||||
|
||||
Widget buildBody(BuildContext context) {
|
||||
final viewModel = ViewModel.of<CentralManagerViewModel>(context);
|
||||
final state = viewModel.state;
|
||||
if (state == BluetoothLowEnergyState.unauthorized) {
|
||||
return Center(
|
||||
child: TextButton(
|
||||
onPressed: () => viewModel.showAppSettings(),
|
||||
child: const Text('Go to settings'),
|
||||
),
|
||||
);
|
||||
} else if (state == BluetoothLowEnergyState.poweredOn) {
|
||||
final discoveries = viewModel.discoveries;
|
||||
return ListView.separated(
|
||||
itemBuilder: (context, index) {
|
||||
final theme = Theme.of(context);
|
||||
final discovery = discoveries[index];
|
||||
final uuid = discovery.peripheral.uuid;
|
||||
final name = discovery.advertisement.name;
|
||||
final rssi = discovery.rssi;
|
||||
return ListTile(
|
||||
onTap: () {
|
||||
onTapDissovery(context, discovery);
|
||||
},
|
||||
onLongPress: () {
|
||||
onLongPressDiscovery(context, discovery);
|
||||
},
|
||||
title: Text(name ?? ''),
|
||||
subtitle: Text(
|
||||
'$uuid',
|
||||
style: theme.textTheme.bodySmall,
|
||||
softWrap: false,
|
||||
maxLines: 1,
|
||||
overflow: TextOverflow.ellipsis,
|
||||
),
|
||||
trailing: Row(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
RSSIIndicator(rssi),
|
||||
Text('$rssi'),
|
||||
],
|
||||
),
|
||||
);
|
||||
},
|
||||
separatorBuilder: (context, i) {
|
||||
return const Divider(
|
||||
height: 0.0,
|
||||
);
|
||||
},
|
||||
itemCount: discoveries.length,
|
||||
);
|
||||
} else {
|
||||
return Center(
|
||||
child: Text(
|
||||
'$state',
|
||||
style: Theme.of(context).textTheme.titleMedium,
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
void onTapDissovery(
|
||||
BuildContext context, DiscoveredEventArgs discovery) async {
|
||||
final viewModel = ViewModel.of<CentralManagerViewModel>(context);
|
||||
if (viewModel.discovering) {
|
||||
await viewModel.stopDiscovery();
|
||||
if (!context.mounted) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
final uuid = discovery.peripheral.uuid;
|
||||
context.go('/central/$uuid');
|
||||
}
|
||||
|
||||
void onLongPressDiscovery(
|
||||
BuildContext context, DiscoveredEventArgs discovery) async {
|
||||
await showModalBottomSheet(
|
||||
context: context,
|
||||
builder: (context) {
|
||||
return AdvertisementView(
|
||||
advertisement: discovery.advertisement,
|
||||
);
|
||||
},
|
||||
);
|
||||
}
|
||||
}
|
@ -0,0 +1,18 @@
|
||||
import 'package:bluetooth_low_energy_android_example/view_models.dart';
|
||||
import 'package:clover/clover.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
class CharacteristicTreeNodeView extends StatelessWidget {
|
||||
const CharacteristicTreeNodeView({super.key});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final viewModel = ViewModel.of<CharacteristicViewModel>(context);
|
||||
return Text(
|
||||
'${viewModel.uuid}',
|
||||
style: Theme.of(context).textTheme.bodyMedium?.copyWith(
|
||||
color: Theme.of(context).colorScheme.secondary,
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
@ -0,0 +1,178 @@
|
||||
import 'dart:convert';
|
||||
|
||||
import 'package:bluetooth_low_energy_platform_interface/bluetooth_low_energy_platform_interface.dart';
|
||||
import 'package:bluetooth_low_energy_android_example/view_models.dart';
|
||||
import 'package:clover/clover.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:material_symbols_icons/material_symbols_icons.dart';
|
||||
|
||||
import 'log_view.dart';
|
||||
|
||||
class CharacteristicView extends StatefulWidget {
|
||||
const CharacteristicView({super.key});
|
||||
|
||||
@override
|
||||
State<CharacteristicView> createState() => _CharacteristicViewState();
|
||||
}
|
||||
|
||||
class _CharacteristicViewState extends State<CharacteristicView> {
|
||||
late final TextEditingController _textController;
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
_textController = TextEditingController();
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final viewModel = ViewModel.of<CharacteristicViewModel>(context);
|
||||
final logs = viewModel.logs;
|
||||
return Column(
|
||||
children: [
|
||||
Expanded(
|
||||
child: LayoutBuilder(
|
||||
builder: (context, constraints) {
|
||||
return OverflowBox(
|
||||
alignment: Alignment.topCenter,
|
||||
maxHeight: constraints.maxHeight + 1.0,
|
||||
child: Row(
|
||||
children: [
|
||||
Expanded(
|
||||
child: InputDecorator(
|
||||
decoration: InputDecoration(
|
||||
contentPadding: const EdgeInsets.all(12.0),
|
||||
border: OutlineInputBorder(
|
||||
borderRadius: BorderRadius.vertical(
|
||||
top: const Radius.circular(12.0),
|
||||
bottom: viewModel.canWrite ||
|
||||
viewModel.canWriteWithoutResponse
|
||||
? Radius.zero
|
||||
: const Radius.circular(12.0),
|
||||
),
|
||||
),
|
||||
),
|
||||
child: ListView.builder(
|
||||
itemBuilder: (context, i) {
|
||||
final log = logs[i];
|
||||
return LogView(
|
||||
log: log,
|
||||
);
|
||||
},
|
||||
itemCount: logs.length,
|
||||
),
|
||||
),
|
||||
),
|
||||
Column(
|
||||
children: [
|
||||
Visibility(
|
||||
visible: viewModel.canNotify,
|
||||
child: viewModel.notifyState
|
||||
? IconButton.filled(
|
||||
onPressed: () async {
|
||||
await viewModel.setNotifyState(false);
|
||||
},
|
||||
icon:
|
||||
const Icon(Symbols.notifications_active),
|
||||
)
|
||||
: IconButton.filledTonal(
|
||||
onPressed: () async {
|
||||
await viewModel.setNotifyState(true);
|
||||
},
|
||||
icon: const Icon(Symbols.notifications_off),
|
||||
),
|
||||
),
|
||||
Visibility(
|
||||
visible: viewModel.canRead,
|
||||
child: IconButton.filled(
|
||||
onPressed: () async {
|
||||
await viewModel.read();
|
||||
},
|
||||
icon: const Icon(Symbols.arrow_downward),
|
||||
),
|
||||
),
|
||||
Visibility(
|
||||
visible: viewModel.canWrite ||
|
||||
viewModel.canWriteWithoutResponse,
|
||||
child: viewModel.writeType ==
|
||||
GATTCharacteristicWriteType.withResponse
|
||||
? IconButton.filled(
|
||||
onPressed: viewModel.canWriteWithoutResponse
|
||||
? () {
|
||||
viewModel.setWriteType(
|
||||
GATTCharacteristicWriteType
|
||||
.withoutResponse);
|
||||
}
|
||||
: null,
|
||||
icon: const Icon(Symbols.swap_vert),
|
||||
)
|
||||
: IconButton.filledTonal(
|
||||
onPressed: viewModel.canWrite
|
||||
? () {
|
||||
viewModel.setWriteType(
|
||||
GATTCharacteristicWriteType
|
||||
.withResponse);
|
||||
}
|
||||
: null,
|
||||
icon: const Icon(Symbols.arrow_upward),
|
||||
),
|
||||
),
|
||||
IconButton.filled(
|
||||
onPressed: () => viewModel.clearLogs(),
|
||||
icon: const Icon(Symbols.delete),
|
||||
),
|
||||
],
|
||||
),
|
||||
],
|
||||
),
|
||||
);
|
||||
},
|
||||
),
|
||||
),
|
||||
Visibility(
|
||||
visible: viewModel.canWrite || viewModel.canWriteWithoutResponse,
|
||||
child: Row(
|
||||
children: [
|
||||
Expanded(
|
||||
child: TextField(
|
||||
controller: _textController,
|
||||
decoration: const InputDecoration(
|
||||
contentPadding: EdgeInsets.symmetric(
|
||||
horizontal: 12.0,
|
||||
),
|
||||
border: OutlineInputBorder(
|
||||
borderRadius: BorderRadius.vertical(
|
||||
bottom: Radius.circular(12.0),
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
ValueListenableBuilder(
|
||||
valueListenable: _textController,
|
||||
builder: (context, tev, child) {
|
||||
final text = tev.text;
|
||||
return IconButton.filled(
|
||||
onPressed: text.isEmpty
|
||||
? null
|
||||
: () async {
|
||||
final value = utf8.encode(text);
|
||||
await viewModel.write(value);
|
||||
},
|
||||
icon: const Icon(Symbols.pets),
|
||||
);
|
||||
},
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
],
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
_textController.dispose();
|
||||
super.dispose();
|
||||
}
|
||||
}
|
@ -0,0 +1,18 @@
|
||||
import 'package:bluetooth_low_energy_android_example/view_models.dart';
|
||||
import 'package:clover/clover.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
class DescriptorTreeNodeView extends StatelessWidget {
|
||||
const DescriptorTreeNodeView({super.key});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final viewModel = ViewModel.of<DescriptorViewModel>(context);
|
||||
return Text(
|
||||
'${viewModel.uuid}',
|
||||
style: Theme.of(context).textTheme.bodySmall?.copyWith(
|
||||
color: Theme.of(context).colorScheme.tertiary,
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
@ -0,0 +1,96 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:go_router/go_router.dart';
|
||||
|
||||
class HomeView extends StatefulWidget {
|
||||
final StatefulNavigationShell navigationShell;
|
||||
final List<Widget> navigators;
|
||||
|
||||
const HomeView({
|
||||
super.key,
|
||||
required this.navigationShell,
|
||||
required this.navigators,
|
||||
});
|
||||
|
||||
@override
|
||||
State<HomeView> createState() => _HomeViewState();
|
||||
}
|
||||
|
||||
class _HomeViewState extends State<HomeView> {
|
||||
late final PageController _controller;
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
_controller = PageController(
|
||||
initialPage: widget.navigationShell.currentIndex,
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final navigationShell = widget.navigationShell;
|
||||
final navigators = widget.navigators;
|
||||
return Scaffold(
|
||||
// body: navigationShell,
|
||||
body: PageView.builder(
|
||||
controller: _controller,
|
||||
onPageChanged: (index) {
|
||||
// Ignore tap events.
|
||||
if (index == navigationShell.currentIndex) {
|
||||
return;
|
||||
}
|
||||
navigationShell.goBranch(
|
||||
index,
|
||||
initialLocation: false,
|
||||
);
|
||||
},
|
||||
itemBuilder: (context, index) {
|
||||
return navigators[index];
|
||||
},
|
||||
itemCount: navigators.length,
|
||||
),
|
||||
bottomNavigationBar: BottomNavigationBar(
|
||||
onTap: (index) {
|
||||
navigationShell.goBranch(
|
||||
index,
|
||||
initialLocation: index == navigationShell.currentIndex,
|
||||
);
|
||||
},
|
||||
items: const [
|
||||
BottomNavigationBarItem(
|
||||
icon: Icon(Icons.radar),
|
||||
label: 'Central',
|
||||
),
|
||||
BottomNavigationBarItem(
|
||||
icon: Icon(Icons.sensors),
|
||||
label: 'Peripheral',
|
||||
),
|
||||
],
|
||||
currentIndex: widget.navigationShell.currentIndex,
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
void didUpdateWidget(covariant HomeView oldWidget) {
|
||||
super.didUpdateWidget(oldWidget);
|
||||
final navigationShell = widget.navigationShell;
|
||||
final page = _controller.page ?? _controller.initialPage;
|
||||
final index = page.round();
|
||||
// Ignore swipe events.
|
||||
if (index == navigationShell.currentIndex) {
|
||||
return;
|
||||
}
|
||||
_controller.animateToPage(
|
||||
navigationShell.currentIndex,
|
||||
duration: const Duration(milliseconds: 300),
|
||||
curve: Curves.linear,
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
_controller.dispose();
|
||||
super.dispose();
|
||||
}
|
||||
}
|
43
bluetooth_low_energy_android/example/lib/views/log_view.dart
Normal file
43
bluetooth_low_energy_android/example/lib/views/log_view.dart
Normal file
@ -0,0 +1,43 @@
|
||||
import 'package:bluetooth_low_energy_android_example/models.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:intl/intl.dart';
|
||||
|
||||
class LogView extends StatelessWidget {
|
||||
final Log log;
|
||||
|
||||
const LogView({
|
||||
super.key,
|
||||
required this.log,
|
||||
});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final formatter = DateFormat.Hms();
|
||||
final time = formatter.format(log.time);
|
||||
return Column(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text.rich(
|
||||
TextSpan(
|
||||
text: '[$time] ',
|
||||
style: Theme.of(context).textTheme.titleSmall?.copyWith(
|
||||
color: Theme.of(context).colorScheme.error,
|
||||
),
|
||||
children: [
|
||||
TextSpan(
|
||||
text: log.type,
|
||||
style: Theme.of(context).textTheme.titleSmall?.copyWith(
|
||||
color: Theme.of(context).colorScheme.primary,
|
||||
)),
|
||||
],
|
||||
),
|
||||
),
|
||||
Text(
|
||||
log.message,
|
||||
style: Theme.of(context).textTheme.bodySmall,
|
||||
),
|
||||
],
|
||||
);
|
||||
}
|
||||
}
|
@ -0,0 +1,76 @@
|
||||
import 'package:bluetooth_low_energy_platform_interface/bluetooth_low_energy_platform_interface.dart';
|
||||
import 'package:bluetooth_low_energy_android_example/view_models.dart';
|
||||
import 'package:clover/clover.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:material_symbols_icons/material_symbols_icons.dart';
|
||||
|
||||
import 'log_view.dart';
|
||||
|
||||
class PeripheralManagerView extends StatelessWidget {
|
||||
const PeripheralManagerView({super.key});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final viewModel = ViewModel.of<PeripheralManagerViewModel>(context);
|
||||
final state = viewModel.state;
|
||||
final advertising = viewModel.advertising;
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: const Text('Peripheral Manager'),
|
||||
actions: [
|
||||
TextButton(
|
||||
onPressed: state == BluetoothLowEnergyState.poweredOn
|
||||
? () async {
|
||||
if (advertising) {
|
||||
await viewModel.stopAdvertising();
|
||||
} else {
|
||||
await viewModel.startAdvertising();
|
||||
}
|
||||
}
|
||||
: null,
|
||||
child: Text(advertising ? 'END' : 'BEGIN'),
|
||||
),
|
||||
],
|
||||
),
|
||||
body: buildBody(context),
|
||||
floatingActionButton: state == BluetoothLowEnergyState.poweredOn
|
||||
? FloatingActionButton(
|
||||
onPressed: () => viewModel.clearLogs(),
|
||||
child: const Icon(Symbols.delete),
|
||||
)
|
||||
: null,
|
||||
);
|
||||
}
|
||||
|
||||
Widget buildBody(BuildContext context) {
|
||||
final viewModel = ViewModel.of<PeripheralManagerViewModel>(context);
|
||||
final state = viewModel.state;
|
||||
if (state == BluetoothLowEnergyState.unauthorized) {
|
||||
return Center(
|
||||
child: TextButton(
|
||||
onPressed: () => viewModel.showAppSettings(),
|
||||
child: const Text('Go to settings'),
|
||||
),
|
||||
);
|
||||
} else if (state == BluetoothLowEnergyState.poweredOn) {
|
||||
final logs = viewModel.logs;
|
||||
return ListView.builder(
|
||||
padding: const EdgeInsets.all(16.0),
|
||||
itemBuilder: (context, i) {
|
||||
final log = logs[i];
|
||||
return LogView(
|
||||
log: log,
|
||||
);
|
||||
},
|
||||
itemCount: logs.length,
|
||||
);
|
||||
} else {
|
||||
return Center(
|
||||
child: Text(
|
||||
'$state',
|
||||
style: Theme.of(context).textTheme.titleMedium,
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,107 @@
|
||||
import 'package:bluetooth_low_energy_android_example/view_models.dart';
|
||||
import 'package:clover/clover.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_simple_treeview/flutter_simple_treeview.dart';
|
||||
|
||||
import 'characteristic_tree_node_view.dart';
|
||||
import 'characteristic_view.dart';
|
||||
import 'descriptor_tree_node_view.dart';
|
||||
import 'service_tree_node_view.dart';
|
||||
|
||||
class PeripheralView extends StatelessWidget {
|
||||
final TreeController _treeController;
|
||||
|
||||
PeripheralView({super.key})
|
||||
: _treeController = TreeController(
|
||||
allNodesExpanded: false,
|
||||
);
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final viewModel = ViewModel.of<PeripheralViewModel>(context);
|
||||
final connected = viewModel.connected;
|
||||
final serviceViewModels = viewModel.serviceViewModels;
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(viewModel.name ?? '${viewModel.uuid}'),
|
||||
actions: [
|
||||
TextButton(
|
||||
onPressed: () async {
|
||||
if (connected) {
|
||||
await viewModel.disconnect();
|
||||
} else {
|
||||
await viewModel.connect();
|
||||
await viewModel.discoverGATT();
|
||||
}
|
||||
},
|
||||
child: Text(connected ? 'DISCONNECT' : 'CONNECT'),
|
||||
),
|
||||
],
|
||||
),
|
||||
body: SingleChildScrollView(
|
||||
child: TreeView(
|
||||
treeController: _treeController,
|
||||
indent: 0.0,
|
||||
nodes: _buildServiceTreeNodes(serviceViewModels),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
List<TreeNode> _buildServiceTreeNodes(List<ServiceViewModel> viewModels) {
|
||||
return viewModels.map((viewModel) {
|
||||
final includedServiceViewModels = viewModel.includedServiceViewModels;
|
||||
final characteristicViewModels = viewModel.characteristicViewModels;
|
||||
return TreeNode(
|
||||
children: [
|
||||
..._buildServiceTreeNodes(includedServiceViewModels),
|
||||
..._buildCharacteristicTreeNodes(characteristicViewModels),
|
||||
],
|
||||
content: InheritedViewModel(
|
||||
view: const ServiceTreeNodeView(),
|
||||
viewModel: viewModel,
|
||||
),
|
||||
);
|
||||
}).toList();
|
||||
}
|
||||
|
||||
List<TreeNode> _buildCharacteristicTreeNodes(
|
||||
List<CharacteristicViewModel> viewModels) {
|
||||
return viewModels.map((viewModel) {
|
||||
final descriptorViewModels = viewModel.descriptorViewModels;
|
||||
return TreeNode(
|
||||
children: [
|
||||
TreeNode(
|
||||
content: Expanded(
|
||||
child: Container(
|
||||
margin: const EdgeInsets.only(right: 40.0),
|
||||
height: 360.0,
|
||||
child: InheritedViewModel(
|
||||
view: const CharacteristicView(),
|
||||
viewModel: viewModel,
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
..._buildDescriptorTreeNodes(descriptorViewModels),
|
||||
],
|
||||
content: InheritedViewModel(
|
||||
view: const CharacteristicTreeNodeView(),
|
||||
viewModel: viewModel,
|
||||
),
|
||||
);
|
||||
}).toList();
|
||||
}
|
||||
|
||||
List<TreeNode> _buildDescriptorTreeNodes(
|
||||
List<DescriptorViewModel> viewModels) {
|
||||
return viewModels.map((viewModel) {
|
||||
return TreeNode(
|
||||
content: InheritedViewModel(
|
||||
view: const DescriptorTreeNodeView(),
|
||||
viewModel: viewModel,
|
||||
),
|
||||
);
|
||||
}).toList();
|
||||
}
|
||||
}
|
@ -0,0 +1,18 @@
|
||||
import 'package:bluetooth_low_energy_android_example/view_models.dart';
|
||||
import 'package:clover/clover.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
class ServiceTreeNodeView extends StatelessWidget {
|
||||
const ServiceTreeNodeView({super.key});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final viewModel = ViewModel.of<ServiceViewModel>(context);
|
||||
return Text(
|
||||
'${viewModel.uuid}',
|
||||
style: Theme.of(context).textTheme.titleSmall?.copyWith(
|
||||
color: Theme.of(context).colorScheme.primary,
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
1
bluetooth_low_energy_android/example/lib/widgets.dart
Normal file
1
bluetooth_low_energy_android/example/lib/widgets.dart
Normal file
@ -0,0 +1 @@
|
||||
export 'widgets/rssi_indicator.dart';
|
@ -0,0 +1,23 @@
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
class RSSIIndicator extends StatelessWidget {
|
||||
final int rssi;
|
||||
|
||||
const RSSIIndicator(
|
||||
this.rssi, {
|
||||
super.key,
|
||||
});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final IconData icon;
|
||||
if (rssi > -70) {
|
||||
icon = Icons.wifi_rounded;
|
||||
} else if (rssi > -100) {
|
||||
icon = Icons.wifi_2_bar_rounded;
|
||||
} else {
|
||||
icon = Icons.wifi_1_bar_rounded;
|
||||
}
|
||||
return Icon(icon);
|
||||
}
|
||||
}
|
@ -15,15 +15,15 @@ packages:
|
||||
path: ".."
|
||||
relative: true
|
||||
source: path
|
||||
version: "5.0.4"
|
||||
version: "6.0.0"
|
||||
bluetooth_low_energy_platform_interface:
|
||||
dependency: "direct main"
|
||||
description:
|
||||
name: bluetooth_low_energy_platform_interface
|
||||
sha256: "5af74eb8f97a896dfdbcff2da284d4fe5b4e2e49ebb2f46f826ac1810f66e4d7"
|
||||
sha256: bc2e8d97c141653e5747bcb3cdc9fe956541b6ecc6e5f158b99a2f3abc2d946a
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "5.0.2"
|
||||
version: "6.0.0"
|
||||
boolean_selector:
|
||||
dependency: transitive
|
||||
description:
|
||||
@ -48,8 +48,16 @@ packages:
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "1.1.1"
|
||||
clover:
|
||||
dependency: "direct main"
|
||||
description:
|
||||
name: clover
|
||||
sha256: eba28e69b32f174a51c093eef098cf5ae646470322727081d5d3d8f66c786487
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "3.0.0"
|
||||
collection:
|
||||
dependency: transitive
|
||||
dependency: "direct main"
|
||||
description:
|
||||
name: collection
|
||||
sha256: ee67cb0715911d28db6bf4af1026078bd6f0128b07a5f66fb2ed94ec6783c09a
|
||||
@ -68,10 +76,10 @@ packages:
|
||||
dependency: "direct main"
|
||||
description:
|
||||
name: cupertino_icons
|
||||
sha256: d57953e10f9f8327ce64a508a355f0b1ec902193f66288e8cb5070e7c47eeb2d
|
||||
sha256: ba631d1c7f7bef6b729a622b7b752645a2d076dba9976925b8f25725a30e1ee6
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "1.0.6"
|
||||
version: "1.0.8"
|
||||
fake_async:
|
||||
dependency: transitive
|
||||
description:
|
||||
@ -102,7 +110,15 @@ packages:
|
||||
dependency: "direct dev"
|
||||
description:
|
||||
name: flutter_lints
|
||||
sha256: "9e8c3858111da373efc5aa341de011d9bd23e2c5c5e0c62bccf32438e192d7b1"
|
||||
sha256: "3f41d009ba7172d5ff9be5f6e6e6abb4300e263aab8866d2a0842ed2a70f8f0c"
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "4.0.0"
|
||||
flutter_simple_treeview:
|
||||
dependency: "direct main"
|
||||
description:
|
||||
name: flutter_simple_treeview
|
||||
sha256: ad4978d2668dd078d3a09966832da111bef9102dd636e572c50c80133b7ff4d9
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "3.0.2"
|
||||
@ -111,11 +127,32 @@ packages:
|
||||
description: flutter
|
||||
source: sdk
|
||||
version: "0.0.0"
|
||||
flutter_web_plugins:
|
||||
dependency: transitive
|
||||
description: flutter
|
||||
source: sdk
|
||||
version: "0.0.0"
|
||||
fuchsia_remote_debug_protocol:
|
||||
dependency: transitive
|
||||
description: flutter
|
||||
source: sdk
|
||||
version: "0.0.0"
|
||||
go_router:
|
||||
dependency: "direct main"
|
||||
description:
|
||||
name: go_router
|
||||
sha256: "6ad5662b014c06c20fa46ab78715c96b2222a7fe4f87bf77e0289592c2539e86"
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "14.1.3"
|
||||
hybrid_logging:
|
||||
dependency: "direct main"
|
||||
description:
|
||||
name: hybrid_logging
|
||||
sha256: "54248d52ce68c14702a42fbc4083bac5c6be30f6afad8a41be4bbadd197b8af5"
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "1.0.0"
|
||||
integration_test:
|
||||
dependency: "direct dev"
|
||||
description: flutter
|
||||
@ -133,44 +170,36 @@ packages:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: leak_tracker
|
||||
sha256: "78eb209deea09858f5269f5a5b02be4049535f568c07b275096836f01ea323fa"
|
||||
sha256: "7f0df31977cb2c0b88585095d168e689669a2cc9b97c309665e3386f3e9d341a"
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "10.0.0"
|
||||
version: "10.0.4"
|
||||
leak_tracker_flutter_testing:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: leak_tracker_flutter_testing
|
||||
sha256: b46c5e37c19120a8a01918cfaf293547f47269f7cb4b0058f21531c2465d6ef0
|
||||
sha256: "06e98f569d004c1315b991ded39924b21af84cf14cc94791b8aea337d25b57f8"
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "2.0.1"
|
||||
version: "3.0.3"
|
||||
leak_tracker_testing:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: leak_tracker_testing
|
||||
sha256: a597f72a664dbd293f3bfc51f9ba69816f84dcd403cdac7066cb3f6003f3ab47
|
||||
sha256: "6ba465d5d76e67ddf503e1161d1f4a6bc42306f9d66ca1e8f079a47290fb06d3"
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "2.0.1"
|
||||
version: "3.0.1"
|
||||
lints:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: lints
|
||||
sha256: cbf8d4b858bb0134ef3ef87841abdf8d63bfc255c266b7bf6b39daa1085c4290
|
||||
sha256: "976c774dd944a42e83e2467f4cc670daef7eed6295b10b36ae8c85bcbf828235"
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "3.0.0"
|
||||
log_service:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: log_service
|
||||
sha256: "21124936899e227d1779268077921d46d57456e2592d1562e455be273594e2e4"
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "1.0.0"
|
||||
version: "4.0.0"
|
||||
logging:
|
||||
dependency: transitive
|
||||
dependency: "direct main"
|
||||
description:
|
||||
name: logging
|
||||
sha256: "623a88c9594aa774443aa3eb2d41807a48486b5613e67599fb4c41c0ad47c340"
|
||||
@ -193,14 +222,22 @@ packages:
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "0.8.0"
|
||||
material_symbols_icons:
|
||||
dependency: "direct main"
|
||||
description:
|
||||
name: material_symbols_icons
|
||||
sha256: b2d3cbc3c42b8a217715b0d97ff03aebb14b2b4592875736e5599c603fb2db7e
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "4.2758.0"
|
||||
meta:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: meta
|
||||
sha256: d584fa6707a52763a52446f02cc621b077888fb63b93bbcb1143a7be5a0c0c04
|
||||
sha256: "7687075e408b093f36e6bbf6c91878cc0d4cd10f409506f7bc996f68220b9136"
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "1.11.0"
|
||||
version: "1.12.0"
|
||||
path:
|
||||
dependency: transitive
|
||||
description:
|
||||
@ -290,10 +327,10 @@ packages:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: test_api
|
||||
sha256: "5c2f730018264d276c20e4f1503fd1308dfbbae39ec8ee63c5236311ac06954b"
|
||||
sha256: "9955ae474176f7ac8ee4e989dadfb411a58c30415bcfb648fa04b2b8a03afa7f"
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "0.6.1"
|
||||
version: "0.7.0"
|
||||
typed_data:
|
||||
dependency: transitive
|
||||
description:
|
||||
@ -314,10 +351,10 @@ packages:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: vm_service
|
||||
sha256: b3d56ff4341b8f182b96aceb2fa20e3dcb336b9f867bc0eafc0de10f1048e957
|
||||
sha256: "3923c89304b715fb1eb6423f017651664a03bf5f4b29983627c4da791f74a4ec"
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "13.0.0"
|
||||
version: "14.2.1"
|
||||
webdriver:
|
||||
dependency: transitive
|
||||
description:
|
||||
@ -327,5 +364,5 @@ packages:
|
||||
source: hosted
|
||||
version: "3.0.3"
|
||||
sdks:
|
||||
dart: ">=3.2.0-0 <4.0.0"
|
||||
flutter: ">=3.0.0"
|
||||
dart: ">=3.3.0 <4.0.0"
|
||||
flutter: ">=3.18.0-18.0.pre.54"
|
||||
|
@ -1,5 +1,5 @@
|
||||
name: bluetooth_low_energy_example
|
||||
description: Demonstrates how to use the bluetooth_low_energy plugin.
|
||||
name: bluetooth_low_energy_android_example
|
||||
description: "Demonstrates how to use the bluetooth_low_energy_android plugin."
|
||||
# The following line prevents the package from being accidentally published to
|
||||
# pub.dev using `flutter pub publish`. This is preferred for private packages.
|
||||
publish_to: 'none' # Remove this line if you wish to publish to pub.dev
|
||||
@ -17,10 +17,10 @@ dependencies:
|
||||
flutter:
|
||||
sdk: flutter
|
||||
|
||||
bluetooth_low_energy_platform_interface: ^5.0.0
|
||||
bluetooth_low_energy_platform_interface: ^6.0.0
|
||||
bluetooth_low_energy_android:
|
||||
# When depending on this package from a real application you should use:
|
||||
# bluetooth_low_energy: ^x.y.z
|
||||
# bluetooth_low_energy_android: ^x.y.z
|
||||
# See https://dart.dev/tools/pub/dependencies#version-constraints
|
||||
# The example app is bundled with the plugin so we use a path dependency on
|
||||
# the parent directory to use the current plugin's version.
|
||||
@ -28,9 +28,16 @@ dependencies:
|
||||
|
||||
# The following adds the Cupertino Icons font to your application.
|
||||
# Use with the CupertinoIcons class for iOS style icons.
|
||||
cupertino_icons: ^1.0.2
|
||||
convert: ^3.1.1
|
||||
cupertino_icons: ^1.0.6
|
||||
clover: ^3.0.0
|
||||
go_router: ^14.1.3
|
||||
logging: ^1.2.0
|
||||
hybrid_logging: ^1.0.0
|
||||
intl: ^0.19.0
|
||||
collection: ^1.18.0
|
||||
convert: ^3.1.1
|
||||
flutter_simple_treeview: ^3.0.2
|
||||
material_symbols_icons: ^4.2744.0
|
||||
|
||||
dev_dependencies:
|
||||
integration_test:
|
||||
@ -43,7 +50,7 @@ dev_dependencies:
|
||||
# activated in the `analysis_options.yaml` file located at the root of your
|
||||
# package. See that file for information about deactivating specific lint
|
||||
# rules and activating additional ones.
|
||||
flutter_lints: ^3.0.0
|
||||
flutter_lints: ^4.0.0
|
||||
|
||||
# For information on the generic Dart part of this file, see the
|
||||
# following page: https://dart.dev/tools/pub/pubspec
|
||||
|
@ -5,23 +5,4 @@
|
||||
// gestures. You can also use WidgetTester to find child widgets in the widget
|
||||
// tree, read text, and verify that the values of widget properties are correct.
|
||||
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_test/flutter_test.dart';
|
||||
|
||||
import 'package:bluetooth_low_energy_example/main.dart';
|
||||
|
||||
void main() {
|
||||
testWidgets('Verify Platform version', (WidgetTester tester) async {
|
||||
// Build our app and trigger a frame.
|
||||
await tester.pumpWidget(const MyApp());
|
||||
|
||||
// Verify that platform version is retrieved.
|
||||
expect(
|
||||
find.byWidgetPredicate(
|
||||
(Widget widget) => widget is Text &&
|
||||
widget.data!.startsWith('Running on:'),
|
||||
),
|
||||
findsOneWidget,
|
||||
);
|
||||
});
|
||||
}
|
||||
void main() {}
|
||||
|
Reference in New Issue
Block a user