首页 > 解决方案 > 全球密钥().currentState.context 在推送时返回 null

问题描述

我正在使用 Flutter Local Notifications 来管理本地通知。我希望当通知类型为“某些用户加入比赛”时,当通知用户点击通知时,他应该被带到比赛详细信息页面。最好的方法是使用 GlobalKey 作为 navigatorKey 并将其传递给 MaterialApp。但我收到了这个错误。

(PS:这个应用程序包含 MainFrameApp() 内的嵌套导航器,因此,在那里处理对 ContestDetailsPage 的进一步导航。)

我找不到什么原因,上下文为空。我认为使用 ChangeNotifier 的 setState 也可能导致了这个问题。所以我也提供了该代码。

════════ Exception caught by widgets library ═══════════════════════════════════
The method 'push' was called on null.
Receiver: null
Tried calling: push<dynamic>(Instance of 'MaterialPageRoute<dynamic>')
The relevant error-causing widget was
MaterialApp
lib\main.dart:90

main.dart

import 'dart:convert';

import 'package:firebase_dynamic_links/firebase_dynamic_links.dart';
import 'package:flutter/material.dart';
import 'package:hive/hive.dart';
import 'package:hive_flutter/hive_flutter.dart';
import 'package:path_provider/path_provider.dart' as pathProvider;
import 'package:provider/provider.dart';
import 'package:shared_preferences/shared_preferences.dart';
import 'package:sports_private_pool/models/authentication.dart';
import 'package:flutter_local_notifications/flutter_local_notifications.dart'
    as local_notification;

import 'models/person.dart';
import 'screens/main_frame_app.dart';
import 'screens/welcome_screen.dart';

bool userLoggedIn = false;
SharedPreferences preferences;

local_notification.FlutterLocalNotificationsPlugin
    flutterLocalNotificationsPlugin =
    local_notification.FlutterLocalNotificationsPlugin();

final GlobalKey<NavigatorState> navigatorKey = GlobalKey<NavigatorState>();

void main() async {
  WidgetsFlutterBinding.ensureInitialized();

  preferences = await SharedPreferences.getInstance();

  final appDocumentDirectory =
      await pathProvider.getApplicationDocumentsDirectory();
  await Hive.initFlutter(appDocumentDirectory.path);
  Hive.registerAdapter(PersonAdapter());

  await Hive.openBox<dynamic>('userData');
  await Hive.openBox<Person>('user');

  // Flutter Local Notification
  var initializationSettingsAndroid =
      local_notification.AndroidInitializationSettings('@mipmap/ic_launcher');
  var initializationSettingsIOS = local_notification.IOSInitializationSettings(
    requestAlertPermission: true,
    requestBadgePermission: true,
    requestSoundPermission: true,
    onDidReceiveLocalNotification:
        (int id, String title, String body, String payload) async {},
  );
  var initializationSettings = local_notification.InitializationSettings(
    android: initializationSettingsAndroid,
    iOS: initializationSettingsIOS,
  );

  flutterLocalNotificationsPlugin.initialize(
    initializationSettings,
    onSelectNotification: (String payload) async {
      if (payload != null) {
        try {
          Map<String, dynamic> payloadJson = json.decode(payload);
          if (payloadJson['data']['type'] == 'joinContest') {

            Navigator.push(
              navigatorKey.currentState.context,
              MaterialPageRoute(
                builder: (context) => MainFrameApp(
                  defaultPage: 2,
                  contestId: payloadJson['data']['contestId'],
                  matchId: payloadJson['data']['matchId'],
                ),
              ),
            );
          }
        } catch (e) {
          print(e);
        }
      } else {
        print("notification payload: $payload");
      }
    },
  );

  runApp(
    ChangeNotifierProvider(
      create: (context) => Authentication(),
      child: MaterialApp(
        title: 'Envision',
        navigatorKey: navigatorKey,
        theme: ThemeData(accentColor: Colors.black87),
        debugShowCheckedModeBanner: false,
        home: Envision(),
      ),
    ),
  );
}

class Envision extends StatefulWidget {
  @override
  _EnvisionState createState() => _EnvisionState();
}

class _EnvisionState extends State<Envision> {
  Box<Person> userBox;

  @override
  initState() {
    userBox = Hive.box('user');
    userLoggedIn = preferences.getString('email') != null;
    print("EMAIL: ${preferences.getString('email')}");

    Future.delayed(Duration.zero, () async {
      if (userLoggedIn) {
        Provider.of<Authentication>(context, listen: false)
            .login(userBox.get('user'));
      }
    });

    super.initState();
  }

  @override
  Widget build(BuildContext context) {
    return Consumer<Authentication>(
      builder: (context, state, child) {
        if (state.loggedIn) {
          return MainFrameApp();
        }
        return WelcomeScreen();
      },
    );
  }
}

身份验证.dart

import 'package:flutter/foundation.dart';
import 'package:sports_private_pool/core/errors/exceptions.dart';
import 'package:sports_private_pool/models/person.dart';

class Authentication extends ChangeNotifier {
  bool _loggedIn = false;
  Person _user;

  bool get loggedIn => _loggedIn;

  Person get user => _loggedIn ? _user : throw NotFoundException();

  void login(Person user) {
    _loggedIn = true;
    _user = user;
    print("user logged in....");
    notifyListeners();
  }

  void logout() {
    _loggedIn = false;
    print("user logged out...");
    notifyListeners();
  }
}

标签: flutternavigatorflutter-providerflutter-change-notifierflutter-local-notification

解决方案


推荐阅读