Использование Google Fit API во Flutter
Мне нужно, чтобы мое приложение считывало количество шагов из Google Fit. Я использую пакет здоровья 3.05. На данный момент я скопировал пример кода, чтобы проверить, работает ли он, но, к сожалению, это не так. Конечно, я выполнил каждый шаг из этого ридми-пакета. Я установил идентификатор клиента OAuth2, я изменил gradle.properties, как они показаны, и в AndroidManifest.xml я поставил
<uses-permission android:name="android.permission.ACTIVITY_RECOGNITION"/>
. Однако после запуска приложения я не получаю никакого окна разрешений, и когда я нажимаю кнопку для получения данных, я получаю сообщение об ошибке «Авторизация не предоставлена» в консоли. Что мне делать? Спасибо
Вот мой код, который я скопировал, пример пакета формы:
import 'dart:async';
import 'package:flutter/material.dart';
import 'package:health/health.dart';
class DailyStepsScreen extends StatefulWidget {
@override
_DailyStepsScreenState createState() => _DailyStepsScreenState();
}
enum AppState {
DATA_NOT_FETCHED,
FETCHING_DATA,
DATA_READY,
NO_DATA,
AUTH_NOT_GRANTED
}
class _DailyStepsScreenState extends State<DailyStepsScreen> {
List<HealthDataPoint> _healthDataList = [];
AppState _state = AppState.DATA_NOT_FETCHED;
@override
void initState() {
super.initState();
}
Future<void> fetchData() async {
/// Get everything from midnight until now
DateTime startDate = DateTime(2020, 11, 07, 0, 0, 0);
DateTime endDate = DateTime(2025, 11, 07, 23, 59, 59);
HealthFactory health = HealthFactory();
/// Define the types to get.
List<HealthDataType> types = [
HealthDataType.STEPS,
HealthDataType.WEIGHT,
HealthDataType.HEIGHT,
HealthDataType.BLOOD_GLUCOSE,
HealthDataType.DISTANCE_WALKING_RUNNING,
];
setState(() => _state = AppState.FETCHING_DATA);
/// You MUST request access to the data types before reading them
bool accessWasGranted = await health.requestAuthorization(types);
int steps = 0;
if (accessWasGranted) {
try {
/// Fetch new data
List<HealthDataPoint> healthData =
await health.getHealthDataFromTypes(startDate, endDate, types);
/// Save all the new data points
_healthDataList.addAll(healthData);
} catch (e) {
print("Caught exception in getHealthDataFromTypes: $e");
}
/// Filter out duplicates
_healthDataList = HealthFactory.removeDuplicates(_healthDataList);
/// Print the results
_healthDataList.forEach((x) {
print("Data point: $x");
steps += x.value.round();
});
print("Steps: $steps");
/// Update the UI to display the results
setState(() {
_state =
_healthDataList.isEmpty ? AppState.NO_DATA : AppState.DATA_READY;
});
} else {
print("Authorization not granted");
setState(() => _state = AppState.DATA_NOT_FETCHED);
}
}
Widget _contentFetchingData() {
return Column(
mainAxisAlignment: MainAxisAlignment.center,
children: <Widget>[
Container(
padding: EdgeInsets.all(20),
child: CircularProgressIndicator(
strokeWidth: 10,
)),
Text('Fetching data...')
],
);
}
Widget _contentDataReady() {
return ListView.builder(
itemCount: _healthDataList.length,
itemBuilder: (_, index) {
HealthDataPoint p = _healthDataList[index];
return ListTile(
title: Text("${p.typeString}: ${p.value}"),
trailing: Text('${p.unitString}'),
subtitle: Text('${p.dateFrom} - ${p.dateTo}'),
);
});
}
Widget _contentNoData() {
return Text('No Data to show');
}
Widget _contentNotFetched() {
return Text('Press the download button to fetch data');
}
Widget _authorizationNotGranted() {
return Text('''Authorization not given.
For Android please check your OAUTH2 client ID is correct in Google Developer Console.
For iOS check your permissions in Apple Health.''');
}
Widget _content() {
if (_state == AppState.DATA_READY)
return _contentDataReady();
else if (_state == AppState.NO_DATA)
return _contentNoData();
else if (_state == AppState.FETCHING_DATA)
return _contentFetchingData();
else if (_state == AppState.AUTH_NOT_GRANTED)
return _authorizationNotGranted();
return _contentNotFetched();
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: const Text('Plugin example app'),
actions: <Widget>[
IconButton(
icon: Icon(Icons.file_download),
onPressed: () {
fetchData();
},
)
],
),
body: Center(
child: _content(),
)
);
}
}
3 ответа
Шаг 1 использование
health: 3.0.4
Шаг 2 выполните правильную настройку идентификатора клиента OAuth2, загрузите новый google-service.json
Шаг 3 Из Android 10. вам нужно добавить
ACTIVITY_RECOGNITION
для получения разрешения STEP Count в
AndroidManifest.xml
.
<uses-permission android:name="android.permission.ACTIVITY_RECOGNITION" />
Шаг 4 Затем с помощью permission_handler запросите разрешение.
if (Platform.isAndroid) {
final permissionStatus = Permission.activityRecognition.request();
if (await permissionStatus.isDenied ||
await permissionStatus.isPermanentlyDenied) {
showToast(
'activityRecognition permission required to fetch your steps count');
return;
}
}
НАКОНЕЦ-ТО РЕШИЛИ ЭТУ ПРОБЛЕМУ!!
Таким образом, проблема не в версии, которую я использую 3.4.0, но проблема все же решена.
Авторизация не предоставлена. И застрял на экране загрузки
Застрял на экране запроса авторизации
Когда вы создаете свой экран согласия OAuth 2.0, попробуйте добавить как минимум 2 адреса электронной почты в раздел TEST USER и убедитесь, что вы вошли в систему с этих электронных писем.
Добавьте 2 адреса электронной почты в Test User
После этого обязательно подтвердите свое приложение в Google, оно будет работать, пока вы не протестируете свое приложение после его выпуска, оно не будет работать.
Шаг 1 использовать здоровье: 3.0.4
Шаг 2 Добавьте функцию разрешения
Шаг 3 Запустите свою функцию внутри initstate