使用 Flutter Provider 将数据传递到另一个屏幕
Passing data to another screen with Flutter Provider
我正在尝试使用 Provider
将数据传递到另一个屏幕,但似乎我总是传递相同的数据,除非我对列表进行排序然后传递不同的数据(意思是我'我可能通过对列表进行排序来切换索引,这就是它现在传递不同数据的原因)。简而言之,我调用 API,填充列表,也为下一页设置提供程序,单击时我列出了前一屏幕的信息,但问题是我总是显示相同的项目除非我对列表进行排序。这是代码:
正在调用 API 并显示列表:
var posts = <RideData>[];
var streamController = StreamController<List<RideData>>();
@override
void initState() {
_getRideStreamList();
super.initState();
}
_getRideStreamList() async {
await Future.delayed(Duration(seconds: 3));
var _vehicleStreamData = await APICalls.instance.getRides();
var provider = Provider.of<RideStore>(context, listen: false);
posts = await _vehicleStreamData
.map<RideData>((e) => RideData.fromJson(e))
.toList();
streamController.add(posts);
provider.setRideList(posts, notify: false);
}
bool isSwitched = true;
void toggleSwitch(bool value) {
if (isSwitched == false) {
posts.sort((k1, k2) => k1.rideId.compareTo(k2.rideId));
} else {
posts.sort((k1, k2) => k2.rideId.compareTo(k1.rideId));
}
}
@override
Widget build(BuildContext context) {
return Scaffold(
backgroundColor: Colors.white,
body: SafeArea(
child: SingleChildScrollView(
scrollDirection: Axis.vertical,
child: Column(
children: [
TextButton(
child: Text('sort ascending'),
onPressed: () {
setState(() {
toggleSwitch(isSwitched = !isSwitched);
});
}),
Container(
height: 1000,
child: StreamBuilder<List<RideData>>(
initialData: posts,
stream: streamController.stream,
builder: (context, snapshot) {
return ListView.builder(
shrinkWrap: true,
itemCount: snapshot.data.length,
itemBuilder: (context, index) {
return Column(
children: [
Row(
children: [
Padding(
padding: const EdgeInsets.only(left: 15.0),
child: Text(
'Ride #${snapshot.data[index].rideId}',
),
),
FlatButton(
textColor: Colors.blue[700],
minWidth: 0,
child: Text('View'),
onPressed: () {
// here is where I pass the data to the RideInfo screen
Navigator.push(
context,
MaterialPageRoute(
builder: (context) => RideInfo(
rideId: snapshot
.data[index].rideId,
)));
},
),
],
),
Column(
crossAxisAlignment: CrossAxisAlignment.stretch,
children: [
Text(
'${snapshot.data[index].pickupTime}',
),
Text(
'${snapshot.data[index].jobArrived}',
),
],
),
],
);
},
);
}),
),
],
),
),
),
);
}
按下“查看”按钮并将数据传递到另一个屏幕(RideInfo)后:
class RideInfo extends StatefulWidget {
static const String id = 'ride_info_screen';
String rideId;
RideInfo({@required this.rideId});
@override
_RideInfoState createState() => _RideInfoState();
}
class _RideInfoState extends State<RideInfo> {
String rideID = '';
@override
void initState() {
super.initState();
setState(() {
rideID = widget.rideId;
});
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
backgroundColor: Colors.white,
title: Text(
'Ride #$rideID',
),
),
body: SafeArea(
child: SingleChildScrollView(
child: Consumer<RideStore>(
builder: (context, rideStore, child) {
return Column(
children: [
ListView.builder(
shrinkWrap: true,
itemCount: 1,
itemBuilder: (context, index) {
RideData rides = rideStore.getRideByIndex(index);
return Column(
children: [
Expanded(
flex: 2,
child: Column(
children: [
Text(
"PICK UP",
),
// here I display the pickUpTime but it is always the same and I wanted to display the time based on the ID
Text(
'${rides.pickupTime}AM',
),
],
),
),
],
);
}),
],
);
},
),
),
),
);
}
}
当我按下查看单个项目的视图时,数据(在本例中为pickUpTime
)不会改变,但是就像我说的,当我使用排序方法更改列表的顺序时,然后我得到不同的数据。
这是提供者模型:
class RideStore extends ChangeNotifier {
List<RideData> _rideList = [];
List<RideData> get rideList => _rideList;
setRideList(List<RideData> list, {bool notify = true}) {
_rideList = list;
if (notify) notifyListeners();
}
RideData getRideByIndex(int index) => _rideList[index];
int get rideListLength => _rideList.length;
}
如何根据我在“行程信息”屏幕中按下并传递的列表中的 ID 显示正确的信息,使其不会始终返回相同的数据?在此先感谢您的帮助!
违规代码在 RideInfo
:
ListView.builder(
shrinkWrap: true,
itemCount: 1,
itemBuilder: (context, index) {
RideData rides = rideStore.getRideByIndex(index);
index
始终是 1
,因此您始终显示第一个 RideData
。有多种选择可以修复它,例如将 index
甚至 RideData
传递给 RideInfo
构造函数:
class RideInfo extends StatefulWidget {
static const String id = 'ride_info_screen';
String rideId;
final int index;
RideInfo({@required this.rideId, @required this.index, Key key})
: super(key: key) {
和:
RideData rides = rideStore.getRideByIndex(widget.index);
我对代码有一些补充意见。首先,ListView
在 RideInfo
中没有任何作用,因此请将其删除。
其次,无需在父窗体中构造streamController
和使用StreamBuilder
。您的列表在 RideStore
中可用。所以你的父表单可以有:
Widget build(BuildContext context) {
var data = Provider.of<RideStore>(context).rideList;
...
Container(
height: 1000,
child:
// StreamBuilder<List<RideData>>(
// initialData: posts,
// stream: streamController.stream,
// builder: (context, snapshot) {
// return
ListView.builder(
shrinkWrap: true,
itemCount: data.length,
希望这些评论对您有所帮助。
编辑:
编辑代码以使用 FutureBuilder
很简单。首先,使_getRideStreamList
return它读取的数据:
_getRideStreamList() async {
...
return posts;
}
删除 initState
中对 _getRideStreamList
的调用,并将 ListView
包装在调用 _getRideStreamList
的 FutureBuilder
中:
Container(
height: 1000,
child: FutureBuilder(
future: _getRideStreamList(),
builder: (ctx, snapshot) {
if (!snapshot.hasData) {
return Center(child: CircularProgressIndicator());
} else {
var data = snapshot.data;
return ListView.builder(
...
);
}
},
),
),
这会在等待数据时显示CircularProgressIndicator
。
请注意,这是一个快速破解 - 您不想每次重建小部件时都读取数据。所以 _getRideStreamList
可以检查数据是否已经被读取并且只是 return 它而不是重新读取。
我正在尝试使用 Provider
将数据传递到另一个屏幕,但似乎我总是传递相同的数据,除非我对列表进行排序然后传递不同的数据(意思是我'我可能通过对列表进行排序来切换索引,这就是它现在传递不同数据的原因)。简而言之,我调用 API,填充列表,也为下一页设置提供程序,单击时我列出了前一屏幕的信息,但问题是我总是显示相同的项目除非我对列表进行排序。这是代码:
正在调用 API 并显示列表:
var posts = <RideData>[];
var streamController = StreamController<List<RideData>>();
@override
void initState() {
_getRideStreamList();
super.initState();
}
_getRideStreamList() async {
await Future.delayed(Duration(seconds: 3));
var _vehicleStreamData = await APICalls.instance.getRides();
var provider = Provider.of<RideStore>(context, listen: false);
posts = await _vehicleStreamData
.map<RideData>((e) => RideData.fromJson(e))
.toList();
streamController.add(posts);
provider.setRideList(posts, notify: false);
}
bool isSwitched = true;
void toggleSwitch(bool value) {
if (isSwitched == false) {
posts.sort((k1, k2) => k1.rideId.compareTo(k2.rideId));
} else {
posts.sort((k1, k2) => k2.rideId.compareTo(k1.rideId));
}
}
@override
Widget build(BuildContext context) {
return Scaffold(
backgroundColor: Colors.white,
body: SafeArea(
child: SingleChildScrollView(
scrollDirection: Axis.vertical,
child: Column(
children: [
TextButton(
child: Text('sort ascending'),
onPressed: () {
setState(() {
toggleSwitch(isSwitched = !isSwitched);
});
}),
Container(
height: 1000,
child: StreamBuilder<List<RideData>>(
initialData: posts,
stream: streamController.stream,
builder: (context, snapshot) {
return ListView.builder(
shrinkWrap: true,
itemCount: snapshot.data.length,
itemBuilder: (context, index) {
return Column(
children: [
Row(
children: [
Padding(
padding: const EdgeInsets.only(left: 15.0),
child: Text(
'Ride #${snapshot.data[index].rideId}',
),
),
FlatButton(
textColor: Colors.blue[700],
minWidth: 0,
child: Text('View'),
onPressed: () {
// here is where I pass the data to the RideInfo screen
Navigator.push(
context,
MaterialPageRoute(
builder: (context) => RideInfo(
rideId: snapshot
.data[index].rideId,
)));
},
),
],
),
Column(
crossAxisAlignment: CrossAxisAlignment.stretch,
children: [
Text(
'${snapshot.data[index].pickupTime}',
),
Text(
'${snapshot.data[index].jobArrived}',
),
],
),
],
);
},
);
}),
),
],
),
),
),
);
}
按下“查看”按钮并将数据传递到另一个屏幕(RideInfo)后:
class RideInfo extends StatefulWidget {
static const String id = 'ride_info_screen';
String rideId;
RideInfo({@required this.rideId});
@override
_RideInfoState createState() => _RideInfoState();
}
class _RideInfoState extends State<RideInfo> {
String rideID = '';
@override
void initState() {
super.initState();
setState(() {
rideID = widget.rideId;
});
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
backgroundColor: Colors.white,
title: Text(
'Ride #$rideID',
),
),
body: SafeArea(
child: SingleChildScrollView(
child: Consumer<RideStore>(
builder: (context, rideStore, child) {
return Column(
children: [
ListView.builder(
shrinkWrap: true,
itemCount: 1,
itemBuilder: (context, index) {
RideData rides = rideStore.getRideByIndex(index);
return Column(
children: [
Expanded(
flex: 2,
child: Column(
children: [
Text(
"PICK UP",
),
// here I display the pickUpTime but it is always the same and I wanted to display the time based on the ID
Text(
'${rides.pickupTime}AM',
),
],
),
),
],
);
}),
],
);
},
),
),
),
);
}
}
当我按下查看单个项目的视图时,数据(在本例中为pickUpTime
)不会改变,但是就像我说的,当我使用排序方法更改列表的顺序时,然后我得到不同的数据。
这是提供者模型:
class RideStore extends ChangeNotifier {
List<RideData> _rideList = [];
List<RideData> get rideList => _rideList;
setRideList(List<RideData> list, {bool notify = true}) {
_rideList = list;
if (notify) notifyListeners();
}
RideData getRideByIndex(int index) => _rideList[index];
int get rideListLength => _rideList.length;
}
如何根据我在“行程信息”屏幕中按下并传递的列表中的 ID 显示正确的信息,使其不会始终返回相同的数据?在此先感谢您的帮助!
违规代码在 RideInfo
:
ListView.builder(
shrinkWrap: true,
itemCount: 1,
itemBuilder: (context, index) {
RideData rides = rideStore.getRideByIndex(index);
index
始终是 1
,因此您始终显示第一个 RideData
。有多种选择可以修复它,例如将 index
甚至 RideData
传递给 RideInfo
构造函数:
class RideInfo extends StatefulWidget {
static const String id = 'ride_info_screen';
String rideId;
final int index;
RideInfo({@required this.rideId, @required this.index, Key key})
: super(key: key) {
和:
RideData rides = rideStore.getRideByIndex(widget.index);
我对代码有一些补充意见。首先,ListView
在 RideInfo
中没有任何作用,因此请将其删除。
其次,无需在父窗体中构造streamController
和使用StreamBuilder
。您的列表在 RideStore
中可用。所以你的父表单可以有:
Widget build(BuildContext context) {
var data = Provider.of<RideStore>(context).rideList;
...
Container(
height: 1000,
child:
// StreamBuilder<List<RideData>>(
// initialData: posts,
// stream: streamController.stream,
// builder: (context, snapshot) {
// return
ListView.builder(
shrinkWrap: true,
itemCount: data.length,
希望这些评论对您有所帮助。
编辑:
编辑代码以使用 FutureBuilder
很简单。首先,使_getRideStreamList
return它读取的数据:
_getRideStreamList() async {
...
return posts;
}
删除 initState
中对 _getRideStreamList
的调用,并将 ListView
包装在调用 _getRideStreamList
的 FutureBuilder
中:
Container(
height: 1000,
child: FutureBuilder(
future: _getRideStreamList(),
builder: (ctx, snapshot) {
if (!snapshot.hasData) {
return Center(child: CircularProgressIndicator());
} else {
var data = snapshot.data;
return ListView.builder(
...
);
}
},
),
),
这会在等待数据时显示CircularProgressIndicator
。
请注意,这是一个快速破解 - 您不想每次重建小部件时都读取数据。所以 _getRideStreamList
可以检查数据是否已经被读取并且只是 return 它而不是重新读取。