How to set Custom height for Widget in GridView in Flutter?

后端 未结 4 1772
一个人的身影
一个人的身影 2020-12-04 10:48

Even after specifying the height for Container GridView, my code is producing square widgets.

class MyHomePage extends StatefulWidget {
  MyHomePage({Key key         


        
相关标签:
4条回答
  • 2020-12-04 11:36

    The key is the childAspectRatio. This value is use to determine the layout in GridView. In order to get the desired aspect you have to set it to the (itemWidth / itemHeight). The solution would be this:

    class MyHomePage extends StatefulWidget {
      MyHomePage({Key key, this.title}) : super(key: key);
    
      final String title;
    
      @override
      _MyHomePageState createState() => new _MyHomePageState();
    }
    
    class _MyHomePageState extends State<MyHomePage> {
      List<String> widgetList = ['A', 'B', 'C'];
    
      @override
      Widget build(BuildContext context) {
        var size = MediaQuery.of(context).size;
    
        /*24 is for notification bar on Android*/
        final double itemHeight = (size.height - kToolbarHeight - 24) / 2;
        final double itemWidth = size.width / 2;
    
        return new Scaffold(
          appBar: new AppBar(
            title: new Text(widget.title),
          ),
          body: new Container(
            child: new GridView.count(
              crossAxisCount: 2,
              childAspectRatio: (itemWidth / itemHeight),
              controller: new ScrollController(keepScrollOffset: false),
              shrinkWrap: true,
              scrollDirection: Axis.vertical,
              children: widgetList.map((String value) {
                return new Container(
                  color: Colors.green,
                  margin: new EdgeInsets.all(1.0),
                  child: new Center(
                    child: new Text(
                      value,
                      style: new TextStyle(
                        fontSize: 50.0,
                        color: Colors.white,
                      ),
                    ),
                  ),
                );
              }).toList(),
            ),
          ),
        );
      }
    }
    
    0 讨论(0)
  • 2020-12-04 11:39

    It's working for me.

    Example code:

    var _crossAxisSpacing = 8;
    var _screenWidth = MediaQuery.of(context).size.width;
    var _crossAxisCount = 2;
    var _width = ( _screenWidth - ((_crossAxisCount - 1) * _crossAxisSpacing)) / _crossAxisCount;
    var cellHeight = 60;
    var _aspectRatio = _width /cellHeight;
    

    GridView:

             GridView.builder(
                              padding: EdgeInsets.only(
                                  left: 5.0, right: 5.0, top: 10, bottom: 10),
                              shrinkWrap: false,
                              itemCount: searchList.length,
                              gridDelegate:
                                  SliverGridDelegateWithFixedCrossAxisCount(crossAxisCount: _crossAxisCount,childAspectRatio: _aspectRatio),
                              itemBuilder: (context, index) {
                                final item = searchList[index];
                                return Card(
                                  child: ListTile(
                                    title: Text(item.name,maxLines: 1,overflow: TextOverflow.ellipsis),
                                    trailing: Container(
                                      width: 15,
                                      height: 15,
                                      decoration: BoxDecoration(
                                          color: item.isActive == true
                                              ? Theme.of(context).primaryColor
                                              : Colors.red,
                                          borderRadius: BorderRadius.all(
                                              Radius.circular(50))),
                                    ),
                                    onTap: () {
    
                                    },
                                  ),
                                  elevation: 0.5,
                                );
                              },
                            ) 
    
    0 讨论(0)
  • 2020-12-04 11:43

    Few days ago I came here to find a way to dynamically change height when images are loaded from internet and using childAspectRatio cannot do that because its apply to all widget in GridView(same height for each).

    This answer may help someone who want different height according to each and every widget content:

    I found a package called Flutter Staggered GridView by Romain Rastel. Using this package we can do so many things check examples here.

    To get what we want we can use StaggeredGridView.count() and its property staggeredTiles: and for its value you can map all widget and apply StaggeredTile.fit(2).

    Example code:

    StaggeredGridView.count(
        crossAxisCount: 4, // I only need two card horizontally
        padding: const EdgeInsets.all(2.0),
        children: yourList.map<Widget>((item) {
          //Do you need to go somewhere when you tap on this card, wrap using InkWell and add your route
          return new Card(
            child: Column(
              children: <Widget>[
                 Image.network(item.yourImage),
                 Text(yourList.yourText),//may be the structure of your data is different
              ],
            ),
          );
        }).toList(),
    
        //Here is the place that we are getting flexible/ dynamic card for various images
        staggeredTiles: yourList.map<StaggeredTile>((_) => StaggeredTile.fit(2))
            .toList(),
        mainAxisSpacing: 3.0,
        crossAxisSpacing: 4.0, // add some space
      ),
    );
    

    You can find complete example(copy,paste and run) here.

    0 讨论(0)
  • 2020-12-04 11:43

    crossAxisCount, crossAxisSpacing and screen width determine width, and childAspectRatio determines height.

    I did bit of calculation to figure out relation between them.

    var width = (screenWidth - ((_crossAxisCount - 1) * _crossAxisSpacing)) / _crossAxisCount;
    var height = width / _aspectRatio;
    

    Full example:

    double _crossAxisSpacing = 8, _mainAxisSpacing = 12, _aspectRatio = 2;
    int _crossAxisCount = 2;
    
    @override
    Widget build(BuildContext context) {
      double screenWidth = MediaQuery.of(context).size.width;
    
      var width = (screenWidth - ((_crossAxisCount - 1) * _crossAxisSpacing)) / _crossAxisCount;
      var height = width / _aspectRatio;
    
      return Scaffold(
        body: GridView.builder(
          itemCount: 10,
          itemBuilder: (context, index) => Container(color: Colors.blue[((index) % 9) * 100]),
          gridDelegate: SliverGridDelegateWithFixedCrossAxisCount(
            crossAxisCount: _crossAxisCount,
            crossAxisSpacing: _crossAxisSpacing,
            mainAxisSpacing: _mainAxisSpacing,
            childAspectRatio: _aspectRatio,
          ),
        ),
      );
    }
    
    0 讨论(0)
提交回复
热议问题