#flutter #dart #flutter-provider #flutter-navigation #flutter-widget
#флаттер #dart #флаттер-провайдер #флаттер-навигация #флаттер-виджет
Вопрос:
Я попытался использовать Hero
виджет с ChangeNotifierProvider
новой системой навигации 2.0. Но я сталкиваюсь с проблемами там. Тот же код работает, когда changeNotifier
не используется.
Проблема, с которой я сталкиваюсь, заключается в том, что после открытия второй страницы W2 Текст (т. Е. дочерний элемент героя) с первой страницы исчезает.
Ниже приведен полный код. runApp(HeroTestApp2())
работает как обычно. Но я бы хотел, чтобы первый подход работал. Пожалуйста, помогите.
import 'package:flutter/material.dart';
import 'package:provider/provider.dart';
main(List<String> args) {
runApp(HeroTestApp());
}
class MyModel extends ChangeNotifier {
bool _go = false;
bool get go => _go;
set go(bool go) {
_go = go;
notifyListeners();
}
}
class HeroTestApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return ChangeNotifierProvider<MyModel>(
create: (context) => MyModel(),
builder: (context, _) {
final state = context.watch<MyModel>();
return MaterialApp(
home: Navigator(
pages: [
MaterialPage(child: W1(onTapped: null)),
if (state.go) MaterialPage(child: W2()),
],
onPopPage: (route, result) {
if (!route.didPop(result)) return false;
state.go = false;
return true;
},
observers: [HeroController()],
),
);
},
);
}
}
class HeroTestApp2 extends StatefulWidget {
@override
_HeroTestApp2State createState() => _HeroTestApp2State();
}
class _HeroTestApp2State extends State<HeroTestApp2> {
bool go = false;
@override
Widget build(BuildContext context) {
return MaterialApp(
home: Navigator(
pages: [
MaterialPage(
child: W1(onTapped: () {
setState(() {
go = true;
});
}),
),
if (go) MaterialPage(child: W2()),
],
onPopPage: (route, result) {
if (!route.didPop(result)) return false;
go = false;
return true;
},
observers: [HeroController()],
),
);
}
}
class W1 extends StatelessWidget {
final void Function() onTapped;
const W1({Key key, this.onTapped}) : super(key: key);
@override
Widget build(BuildContext context) {
return GestureDetector(
onTap: onTapped ??
() => context.read<MyModel>().go = true,
child: Scaffold(
appBar: AppBar(),
body: Container(
alignment: Alignment.topRight,
child: Hero(
tag: "topic",
child: DefaultTextStyle(
style: TextStyle(
fontSize: 40,
decoration: TextDecoration.none,
color: Colors.blue),
child: Text(
"data",
style: TextStyle(fontSize: 40),
),
),
),
),
),
);
}
}
class W2 extends StatelessWidget {
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(),
body: Container(
alignment: Alignment.bottomLeft,
child: Hero(
tag: "topic",
child: DefaultTextStyle(
style: TextStyle(
fontSize: 140,
decoration: TextDecoration.none,
color: Colors.blue),
child: Text(
"data",
),
),
),
),
);
}
}
Ответ №1:
Хорошо, я нахожу ответ.
Это был observers: [HeroController()],
Каждый раз, когда состояние меняется, дерево отверстий перестраивалось вместе с HeroController. Решение состоит в том, чтобы определить HeroController над ChangeNotifier:
class HeroTestApp extends StatelessWidget {
final heroC = HeroController();
@override
Widget build(BuildContext context) {
return ChangeNotifierProvider<MyModel>(
create: (context) => MyModel(),
builder: (context, _) {
final state = context.watch<MyModel>();
return MaterialApp(
home: Navigator(
pages: [
MaterialPage(child: W1(onTapped: null)),
if (state.go) MaterialPage(child: W2()),
],
onPopPage: (route, result) {
if (!route.didPop(result)) return false;
state.go = false;
return true;
},
observers: [heroC],
),
);
},
);
}
}
Спасибо.
Комментарии:
1. Ткс, ты спас мой день, братан
2. Приятно слышать, что это кому-то помогло. 👍