如何在flutter中请求位置许可之前显示披露窗口

xe55xuns  于 2023-06-30  发布在  Flutter
关注(0)|答案(1)|浏览(80)

我需要显示一个披露窗口,然后要求位置权限的用户,使应用程序Google Play符合我的Flutter应用程序。我做了以下更改,但之后既没有位置许可窗口,也没有显示带有文本“我们正在收集位置,为您推荐有趣的地方”的披露消息窗口。我用的是最新的flutter版本。以下是相关的类:

import 'dart:async';
import 'package:flutter/material.dart';
import 'package:location/location.dart';

import '../../Datamodel/user_location.dart';

class LocationService {
  static final LocationService _instance = LocationService._internal();

  factory LocationService(BuildContext context) {
    return _instance;
  }
  LocationService._internal({BuildContext? context}) {
    if (context != null) {
    getLocationOnchange(context);
  }
  }

  Location location = Location();
  final StreamController<UserLocation> _locationController =
  StreamController<UserLocation>.broadcast();
  StreamSubscription<LocationData>? listener;

  Stream<UserLocation> get locationStream => _locationController.stream;

  void getLocationOnchange(BuildContext context) async {
    showDialog(
      context: context,
      builder: (BuildContext context) {
        return AlertDialog(
          title: const Text('Location Tracking'),
          content: const Text('We are collecting location for suggesting you interesting places.'),
          actions: [
            TextButton(
              onPressed: () {
                Navigator.pop(context);
                requestLocationPermission();
              },
              child: Text('OK'),
            ),
          ],
        );
      },
    );
  }

  void requestLocationPermission() {
    location.requestService().then((value) {
      location.requestPermission().then((permissionStatus) {
        if (permissionStatus == PermissionStatus.granted) {
          location.enableBackgroundMode(enable: true);
          location.changeSettings(
            interval: 60000,
            accuracy: LocationAccuracy.high,
          );
          listener = initListener();
        }
      });
    });
  }

  StreamSubscription<LocationData>? getListener() {
    return listener;
  }

  StreamSubscription<LocationData> initListener() {
    return location.onLocationChanged.listen((locationData) {
      _locationController.add(UserLocation(
        locationData.latitude,
        locationData.longitude,
        locationData.altitude,
        locationData.time,
      ));
    });
  }

  void dispose() {
    listener?.cancel();
    listener = null;
  }

}

startLocation.dart:

import 'package:flutter/cupertino.dart';
import '../services/location/LocationService.dart';
import '../services/location/post_location.dart';

Future<void> startLocation(BuildContext context) async {
  var listener = LocationService(context).getListener();
  listener ??= LocationService(context).initListener();
  LocationService(context).locationStream;
  await startLocationStream(context);
}

main.dart:

import 'dart:async';
import 'package:easy_localization/easy_localization.dart';
import 'package:firebase_core/firebase_core.dart';
import 'package:firebase_messaging/firebase_messaging.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter_local_notifications/flutter_local_notifications.dart';
import 'package:sdx_patient_portal/backend/services/location/LocationService.dart';
import 'package:sdx_patient_portal/backend/services/sharedPref.dart';
import 'package:sdx_patient_portal/firebase_options.dart';

import 'MyApp.dart';

Future<void> backgroundHandler(RemoteMessage message) async {
  await Firebase.initializeApp(options: DefaultFirebaseOptions.currentPlatform);
}

late AndroidNotificationChannel channel;

late FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin;

void main() async {
  WidgetsFlutterBinding.ensureInitialized();
  await Firebase.initializeApp();
  var messaging = FirebaseMessaging.instance;
  await EasyLocalization.ensureInitialized();
  String routeFromMessage = '';

  BuildContext? appContext;

  runApp(
    Builder(
      builder: (BuildContext context) {
        appContext = context;
        return Container();
      },
    ),
  );

  messaging.getInitialMessage().then((message) {
    if (message != null) {
      routeFromMessage = "/" + message.data["route"];
    }
  });
  await messaging.requestPermission(
    alert: true,
    announcement: false,
    badge: true,
    carPlay: false,
    criticalAlert: false,
    provisional: false,
    sound: true,
  );

  LocationService(appContext!).getListener()?.resume();
  FirebaseMessaging.onBackgroundMessage(backgroundHandler);

  if (!kIsWeb) {
    channel = const AndroidNotificationChannel(
        'high_importance_channel', 'High Importance Notifications',
        importance: Importance.high);

    flutterLocalNotificationsPlugin = FlutterLocalNotificationsPlugin();

    await flutterLocalNotificationsPlugin
        .resolvePlatformSpecificImplementation<
            AndroidFlutterLocalNotificationsPlugin>()
        ?.createNotificationChannel(channel);

    await messaging.setForegroundNotificationPresentationOptions(
      alert: true,
      badge: true,
      sound: true,
    );
  }

  if (defaultTargetPlatform == TargetPlatform.android) {
    messaging.getToken().then((newToken) {
      SharedPref.saveFCMToken(newToken);
    });
  } else if (defaultTargetPlatform == TargetPlatform.iOS) {
    messaging.requestPermission(alert: true, sound: true);
    await messaging.setForegroundNotificationPresentationOptions(
      alert: true,
      badge: true,
      sound: true,
    );
    messaging.getToken().then((newToken) {
      SharedPref.saveFCMToken(newToken);
    });
  }

  bool? isLoggedIn = await SharedPref.getUserLoginSharedPrefernces();
  isLoggedIn ??= false;
  String route = '';
  if (isLoggedIn && routeFromMessage != '') {
    route = routeFromMessage;
  } else {
    route = isLoggedIn ? '/' : '/loginScreen';
  }

  runApp(EasyLocalization(
    supportedLocales: const [Locale('en', 'US'), Locale('es', 'ES'), Locale('nl', 'NL'), Locale('de', 'DE')],
    path: 'i18n',
    fallbackLocale: const Locale('en', 'US'),
    child: MyApp(
          homepage: route,
          isLoggedIn: isLoggedIn,
          isLightTheme: true
        ),
  ));
}

init:

class PushMessaging extends StatefulWidget {
  final String title;
  final String homepage;
  final bool? isLoggedIn;

  const PushMessaging(
      {Key? key, required this.homepage, required this.title, this.isLoggedIn})
      : super(key: key);

  @override
  State<PushMessaging> createState() => _PushMessagingState();
}

class _PushMessagingState extends State<PushMessaging> {
  FirebaseMessaging messaging = FirebaseMessaging.instance;

  bool isLoading = false;
  bool isLightTheme = true;    
  int steps = 0;
  String version = '';

  void initPermission() async {
    NotificationSettings settings = await messaging.requestPermission(
      alert: true,
      announcement: false,
      badge: true,
      carPlay: false,
      criticalAlert: false,
      provisional: true,
      sound: true,
    );

    if (settings.authorizationStatus == AuthorizationStatus.authorized) {
      if (kDebugMode) {
        print('User granted permission');
      }
    } else if (settings.authorizationStatus ==
        AuthorizationStatus.provisional) {
      if (kDebugMode) {
        print('User granted provisional permission');
      }
    } else {
      if (kDebugMode) {
        print('User declined or has not accepted permission');
      }
    }
  }

  void toggleTheme() {
    setState(() => isLightTheme = !isLightTheme);
  }

  @override
  void initState() {
    super.initState();
    getNewVersion();
    getAmountOfQuest();
    PedometerService.instance.init();
    startLocation(context);
    initPermission();
    ...rest of code
fzsnzjdm

fzsnzjdm1#

试试这个:

static showPermissionDialogAndAskPermission(
      BuildContext context,
      Permission permission) {
    return showDialog(
        context: context,
        builder: (BuildContext dialogContext) {
          return AlertDialog(
            shape: RoundedRectangleBorder(
              borderRadius: BorderRadius.circular(2.5.w),
            ),
            title: Row(
              mainAxisAlignment: MainAxisAlignment.spaceBetween,
              children: [
                const Text(
                  "Permission required",
                ),
                GestureDetector(
                    onTap: () {
                      Navigator.pop(dialogContext);
                    },
                    child: Icon(
                      Icons.close,
                      size: 6.w,
                    )),
              ],
            ),
            content: const Text("We are collecting location for suggesting you interesting places."),
            actionsPadding: EdgeInsets.fromLTRB(5.w, 0, 5.w, 5.w),
            actions: [
              Row(
                children: [
                  GestureDetector(
                    onTap: () {
                      Navigator.of(context).pop();
                    },
                    child: const Center(
                      child: Text(
                        "Cancel",
                      ),
                    ),
                  ).expand,
                  GestureDetector(
                    onTap: () async {
                      Navigator.of(context).pop();
                      var status = await permission.request();
                      if (status.isGranted) {
                        // Do whatever you want
                      }
                      else if(status.isDenied){
                        // Do whatever you want
                      }
                      else if(status.isPermanentlyDenied){
                        // Do whatever you want
                      }
                    },
                    child: const Center(
                      child: Text(
                        "Grant",
                      ),
                    ),
                  ).expand
                ],
              )
            ],
            actionsAlignment: MainAxisAlignment.spaceAround,
          );
        });
  }

你应该根据你的需要给予文本样式。

相关问题