我试图做
编写您的第一个Flutter应用程序,第2部分flutter app page 5
我现在对此应用程序有疑问。我想像这样从onLongPress列表中删除一个条目:
onLongPress: () {
setState(() {
_saved.remove(pair);
});
},
这将从列表中删除该项目,但不会更新屏幕。返回家园并重新打开此路线后,新项目将被删除。但是,如何在用户不重新打开页面的情况下触发此页面上的更新。
import 'package:flutter/material.dart';
import 'package:english_words/english_words.dart';
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return new MaterialApp(
title: 'Startup Name Generator',
home: RandomWords(),
theme: new ThemeData(
primaryColor: Colors.orange,
),
);
}
}
class RandomWords extends StatefulWidget {
@override
RandomWordsState createState() => RandomWordsState();
}
class RandomWordsState extends State<RandomWords> {
final List<WordPair> _suggestions = <WordPair>[];
final TextStyle _biggerFont = const TextStyle(fontSize: 18);
final Set<WordPair> _saved = new Set<WordPair>();
_buildSuggestions() {
return ListView.builder(
padding: const EdgeInsets.all(16),
itemBuilder: (context, i) {
if (i.isOdd) return Divider();
final index = i ~/ 2;
if (index >= _suggestions.length) {
_suggestions.addAll(generateWordPairs().take(10));
}
return _buildRow(_suggestions[index]);
},
);
}
Widget _buildRow(WordPair pair) {
final bool alreadySaved = _saved.contains(pair);
return ListTile(
title: Text(pair.asPascalCase, style: _biggerFont),
trailing: new Icon(
alreadySaved ? Icons.favorite : Icons.favorite_border,
color: alreadySaved ? Colors.red : null,
),
onTap: () {
setState(() {
if (alreadySaved) {
_saved.remove(pair);
} else {
_saved.add(pair);
}
});
},
);
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text("Startup Name Generator"),
actions: <Widget>[
new IconButton(icon: const Icon(Icons.list), onPressed: _pushSaved),
],
),
body: _buildSuggestions(),
);
}
void _pushSaved() {
Navigator.of(context).push(
new MaterialPageRoute(
builder: (BuildContext context) {
final Iterable<ListTile> tiles = _saved.map(
(WordPair pair) {
return new ListTile(
//this is the delete operation
onLongPress: () {
setState(() {
_saved.remove(pair);
});
},
title: new Text(
pair.asPascalCase,
style: _biggerFont,
),
);
},
);
final List<Widget> divided = ListTile.divideTiles(
context: context,
tiles: tiles,
).toList();
return new Scaffold(
appBar: new AppBar(
title: const Text('Saved Suggestions'),
),
body: new ListView(children: divided),
);
},
),
);
}
}
答案 0 :(得分:2)
那是因为您正在创建一个新的MaterialPageRoute。
尝试一下:
onLongPress: () {
_saved.remove(pair);
Navigator.of(context).pop();
_pushSaved();
},
使用此解决方案,您仍然会看到视图的变化。如果您想避免这种情况,则需要一个新的有状态页面,并进行一些重构:
_saved
项设为全局(仅在此示例中)_pushSaved
方法onPressed
函数的_pushSaved
函数DetailPage
而不是_pushSaved
方法像这样:
import 'package:flutter/material.dart';
import 'package:english_words/english_words.dart';
void main() => runApp(new MyApp());
// create a global saved set
Set<WordPair> savedGlobal = new Set<WordPair>();
class MyApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return new MaterialApp(
title: 'Startup Name Generator',
home: new RandomWords(),
);
}
}
class RandomWords extends StatefulWidget {
@override
RandomWordsState createState() => new RandomWordsState();
}
class RandomWordsState extends State<RandomWords> {
final List<WordPair> _suggestions = <WordPair>[];
final TextStyle _biggerFont = const TextStyle(fontSize: 18.0);
@override
Widget build(BuildContext context) {
return new Scaffold(
appBar: new AppBar(
title: const Text('Startup Name Generator'),
actions: <Widget>[
// change the onPressed function
new IconButton(icon: const Icon(Icons.list), onPressed: () {
Navigator.push(
context,
MaterialPageRoute(
builder: (context) => DetailPage()
)
);
}),
],
),
body: _buildSuggestions(),
);
}
Widget _buildSuggestions() {
return new ListView.builder(
padding: const EdgeInsets.all(16.0),
itemBuilder: (BuildContext _context, int i) {
if (i.isOdd) {
return const Divider();
}
final int index = i ~/ 2;
if (index >= _suggestions.length) {
_suggestions.addAll(generateWordPairs().take(10));
}
return _buildRow(_suggestions[index]);
});
}
Widget _buildRow(WordPair pair) {
final bool alreadySaved = savedGlobal.contains(pair);
return new ListTile(
title: new Text(
pair.asPascalCase,
style: _biggerFont,
),
trailing: new Icon(
alreadySaved ? Icons.favorite : Icons.favorite_border,
color: alreadySaved ? Colors.red : null,
),
onTap: () {
setState(() {
if (alreadySaved) {
savedGlobal.remove(pair);
} else {
savedGlobal.add(pair);
}
});
},
);
}
}
// add a new stateful page
class DetailPage extends StatefulWidget {
@override
_DetailPageState createState() => _DetailPageState();
}
class _DetailPageState extends State<DetailPage> {
final TextStyle _biggerFont = const TextStyle(fontSize: 18.0);
@override
Widget build(BuildContext context) {
Iterable<ListTile> tiles = savedGlobal.map((WordPair pair) {
return new ListTile(
onLongPress: () {
setState(() {
savedGlobal.remove(pair);
});
},
title: new Text(
pair.asPascalCase,
style: _biggerFont,
),
);
});
final List<Widget> divided = ListTile.divideTiles(
context: context,
tiles: tiles,
).toList();
return new Scaffold(
appBar: new AppBar(
title: const Text('Saved Suggestions'),
),
body: new ListView(children: divided),
);
}
}
答案 1 :(得分:0)
2019年11月最新方式
更多信息https://www.youtube.com/watch?v=iEMgjrfuc58
ListView.builder/GridView.builder(
itemBuilder: (BuildContext context, int index){
return Dismissible(
key: Key(selectServiceLocations[index].toString()),
onDismissed: (direction) {
setState(() {
selectServiceLocations.removeAt(index);
});
},
child: Container(...)
}
)
答案 2 :(得分:0)
聚会晚了一点,但我发现,如果您想操纵ListView或GridView,那么为List / GridView的每个子小部件分配一个键至关重要。
简而言之,Flutter仅按类型而不按状态比较小部件。因此,当List / GridView中表示的List的状态发生更改时,Flutter不知道应删除哪些子级,因为它们的Types仍然相同,然后签出。 Flutter拾取的唯一问题是项目数,这就是为什么它仅删除List / GridView中的最后一个小部件的原因。
因此,如果要在Flutter中操作列表,请为每个孩子的顶层窗口小部件分配一个Key。 this article中提供了更详细的说明。
这可以通过添加
来实现 return GridView.count(
shrinkWrap: true,
crossAxisCount: 2,
crossAxisSpacing: 5.0,
mainAxisSpacing: 5.0,
children: List.generate(urls.length, (index) {
//generating tiles with from list
return GestureDetector(
key: UniqueKey(), //This made all the difference for me
onTap: () => {
setState(() {
currentUrls.removeAt(index);
})
},
child: FadeInImage( // A custom widget I made to display an Image from
image: NetworkImage(urls[index]),
placeholder: AssetImage('assets/error_loading.png')
),
);
}),
);
答案 3 :(得分:0)
第 1 步:查找特定对象(元素)VIA 的索引
List_Name.indexOf(List_Name[index]);
ex:_userTransactions.indexOf(_userTransactions[index])
第 2 步:删除特定索引处的对象
LIST_Name.removeAt(index);
ex: _userTransactions.removeAt(index);