// Copyright 2014 The Flutter Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. import 'package:flutter/material.dart'; import 'package:flutter/scheduler.dart' show timeDilation; enum ScrollMode { complex, tile } class ComplexLayoutApp extends StatefulWidget { const ComplexLayoutApp({super.key, this.badScroll = false}); final bool badScroll; @override ComplexLayoutAppState createState() => ComplexLayoutAppState(); static ComplexLayoutAppState? of(BuildContext context) => context.findAncestorStateOfType<ComplexLayoutAppState>(); } class ComplexLayoutAppState extends State<ComplexLayoutApp> { @override Widget build(BuildContext context) { return MaterialApp( theme: lightTheme ? ThemeData.light() : ThemeData.dark(), title: 'Advanced Layout', home: scrollMode == ScrollMode.complex ? ComplexLayout(badScroll: widget.badScroll) : const TileScrollLayout()); } bool _lightTheme = true; bool get lightTheme => _lightTheme; set lightTheme(bool value) { setState(() { _lightTheme = value; }); } ScrollMode _scrollMode = ScrollMode.complex; ScrollMode get scrollMode => _scrollMode; set scrollMode(ScrollMode mode) { setState(() { _scrollMode = mode; }); } void toggleAnimationSpeed() { setState(() { timeDilation = (timeDilation != 1.0) ? 1.0 : 5.0; }); } } class TileScrollLayout extends StatelessWidget { const TileScrollLayout({ super.key }); @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar(title: const Text('Tile Scrolling Layout')), body: ListView.builder( key: const Key('tiles-scroll'), itemCount: 200, itemBuilder: (BuildContext context, int index) { return Padding( padding: const EdgeInsets.all(5.0), child: Material( elevation: (index % 5 + 1).toDouble(), color: Colors.white, child: const IconBar(), ), ); }, ), drawer: const GalleryDrawer(), ); } } class ComplexLayout extends StatefulWidget { const ComplexLayout({ super.key, required this.badScroll }); final bool badScroll; @override ComplexLayoutState createState() => ComplexLayoutState(); static ComplexLayoutState? of(BuildContext context) => context.findAncestorStateOfType<ComplexLayoutState>(); } class ComplexLayoutState extends State<ComplexLayout> { @override Widget build(BuildContext context) { Widget body = ListView.builder( key: const Key('complex-scroll'), // this key is used by the driver test controller: ScrollController(), // So that the scroll offset can be tracked itemCount: widget.badScroll ? 500 : null, shrinkWrap: widget.badScroll, itemBuilder: (BuildContext context, int index) { if (index.isEven) { return FancyImageItem(index, key: PageStorageKey<int>(index)); } else { return FancyGalleryItem(index, key: PageStorageKey<int>(index)); } }, ); if (widget.badScroll) { body = ListView( key: const Key('complex-scroll-bad'), children: <Widget>[body], ); } return Scaffold( appBar: AppBar( title: const Text('Advanced Layout'), actions: <Widget>[ IconButton( icon: const Icon(Icons.create), tooltip: 'Search', onPressed: () { print('Pressed search'); }, ), const TopBarMenu(), ], ), body: Column( children: <Widget>[ Expanded(child: body), const BottomBar(), ], ), drawer: const GalleryDrawer(), ); } } class TopBarMenu extends StatelessWidget { const TopBarMenu({super.key}); @override Widget build(BuildContext context) { return PopupMenuButton<String>( onSelected: (String value) { print('Selected: $value'); }, itemBuilder: (BuildContext context) => <PopupMenuItem<String>>[ const PopupMenuItem<String>( value: 'Friends', child: MenuItemWithIcon(Icons.people, 'Friends', '5 new'), ), const PopupMenuItem<String>( value: 'Events', child: MenuItemWithIcon(Icons.event, 'Events', '12 upcoming'), ), const PopupMenuItem<String>( value: 'Events', child: MenuItemWithIcon(Icons.group, 'Groups', '14'), ), const PopupMenuItem<String>( value: 'Events', child: MenuItemWithIcon(Icons.image, 'Pictures', '12'), ), const PopupMenuItem<String>( value: 'Events', child: MenuItemWithIcon(Icons.near_me, 'Nearby', '33'), ), const PopupMenuItem<String>( value: 'Friends', child: MenuItemWithIcon(Icons.people, 'Friends', '5'), ), const PopupMenuItem<String>( value: 'Events', child: MenuItemWithIcon(Icons.event, 'Events', '12'), ), const PopupMenuItem<String>( value: 'Events', child: MenuItemWithIcon(Icons.group, 'Groups', '14'), ), const PopupMenuItem<String>( value: 'Events', child: MenuItemWithIcon(Icons.image, 'Pictures', '12'), ), const PopupMenuItem<String>( value: 'Events', child: MenuItemWithIcon(Icons.near_me, 'Nearby', '33'), ), ], ); } } class MenuItemWithIcon extends StatelessWidget { const MenuItemWithIcon(this.icon, this.title, this.subtitle, {super.key}); final IconData icon; final String title; final String subtitle; @override Widget build(BuildContext context) { return Row( children: <Widget>[ Icon(icon), Padding( padding: const EdgeInsets.only(left: 8.0, right: 8.0), child: Text(title), ), Text(subtitle, style: Theme.of(context).textTheme.bodySmall), ], ); } } class FancyImageItem extends StatelessWidget { const FancyImageItem(this.index, {super.key}); final int index; @override Widget build(BuildContext context) { return ListBody( children: <Widget>[ UserHeader('Ali Connors $index'), const ItemDescription(), const ItemImageBox(), const InfoBar(), const Padding( padding: EdgeInsets.symmetric(horizontal: 8.0), child: Divider(), ), const IconBar(), const FatDivider(), ], ); } } class FancyGalleryItem extends StatelessWidget { const FancyGalleryItem(this.index, {super.key}); final int index; @override Widget build(BuildContext context) { return ListBody( children: <Widget>[ const UserHeader('Ali Connors'), ItemGalleryBox(index), const InfoBar(), const Padding( padding: EdgeInsets.symmetric(horizontal: 8.0), child: Divider(), ), const IconBar(), const FatDivider(), ], ); } } class InfoBar extends StatelessWidget { const InfoBar({super.key}); @override Widget build(BuildContext context) { return Padding( padding: const EdgeInsets.all(8.0), child: Row( mainAxisAlignment: MainAxisAlignment.spaceBetween, children: <Widget>[ const MiniIconWithText(Icons.thumb_up, '42'), Text('3 Comments', style: Theme.of(context).textTheme.bodySmall), ], ), ); } } class IconBar extends StatelessWidget { const IconBar({super.key}); @override Widget build(BuildContext context) { return const Padding( padding: EdgeInsets.only(left: 16.0, right: 16.0), child: Row( mainAxisAlignment: MainAxisAlignment.spaceBetween, children: <Widget>[ IconWithText(Icons.thumb_up, 'Like'), IconWithText(Icons.comment, 'Comment'), IconWithText(Icons.share, 'Share'), ], ), ); } } class IconWithText extends StatelessWidget { const IconWithText(this.icon, this.title, {super.key}); final IconData icon; final String title; @override Widget build(BuildContext context) { return Row( mainAxisSize: MainAxisSize.min, children: <Widget>[ IconButton( icon: Icon(icon), onPressed: () { print('Pressed $title button'); }, ), Text(title), ], ); } } class MiniIconWithText extends StatelessWidget { const MiniIconWithText(this.icon, this.title, {super.key}); final IconData icon; final String title; @override Widget build(BuildContext context) { return Row( mainAxisSize: MainAxisSize.min, children: <Widget>[ Padding( padding: const EdgeInsets.only(right: 8.0), child: Container( width: 16.0, height: 16.0, decoration: ShapeDecoration( color: Theme.of(context).primaryColor, shape: const CircleBorder(), ), child: Icon(icon, color: Colors.white, size: 12.0), ), ), Text(title, style: Theme.of(context).textTheme.bodySmall), ], ); } } class FatDivider extends StatelessWidget { const FatDivider({super.key}); @override Widget build(BuildContext context) { return Container( height: 8.0, color: Theme.of(context).dividerColor, ); } } class UserHeader extends StatelessWidget { const UserHeader(this.userName, {super.key}); final String userName; @override Widget build(BuildContext context) { return Padding( padding: const EdgeInsets.all(8.0), child: Row( crossAxisAlignment: CrossAxisAlignment.start, children: <Widget>[ const Padding( padding: EdgeInsets.only(right: 8.0), child: Image( image: AssetImage('packages/flutter_gallery_assets/people/square/ali.png'), width: 32.0, height: 32.0, ), ), Expanded( child: Column( crossAxisAlignment: CrossAxisAlignment.stretch, children: <Widget>[ RichText(text: TextSpan( style: Theme.of(context).textTheme.bodyMedium, children: <TextSpan>[ TextSpan(text: userName, style: const TextStyle(fontWeight: FontWeight.bold)), const TextSpan(text: ' shared a new '), const TextSpan(text: 'photo', style: TextStyle(fontWeight: FontWeight.bold)), ], )), Row( children: <Widget>[ Text('Yesterday at 11:55 • ', style: Theme.of(context).textTheme.bodySmall), Icon(Icons.people, size: 16.0, color: Theme.of(context).textTheme.bodySmall!.color), ], ), ], ), ), const TopBarMenu(), ], ), ); } } class ItemDescription extends StatelessWidget { const ItemDescription({super.key}); @override Widget build(BuildContext context) { return const Padding( padding: EdgeInsets.all(8.0), child: Text('Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.'), ); } } class ItemImageBox extends StatelessWidget { const ItemImageBox({super.key}); @override Widget build(BuildContext context) { return Padding( padding: const EdgeInsets.all(8.0), child: Card( child: Column( crossAxisAlignment: CrossAxisAlignment.stretch, children: <Widget>[ Stack( children: <Widget>[ const SizedBox( height: 230.0, child: Image( image: AssetImage('packages/flutter_gallery_assets/places/india_chettinad_silk_maker.png') ), ), Theme( data: ThemeData.dark(), child: Row( mainAxisAlignment: MainAxisAlignment.end, children: <Widget>[ IconButton( icon: const Icon(Icons.edit), onPressed: () { print('Pressed edit button'); }, ), IconButton( icon: const Icon(Icons.zoom_in), onPressed: () { print('Pressed zoom button'); }, ), ], ), ), Positioned( bottom: 4.0, left: 4.0, child: Container( decoration: BoxDecoration( color: Colors.black54, borderRadius: BorderRadius.circular(2.0), ), padding: const EdgeInsets.all(4.0), child: RichText( text: const TextSpan( style: TextStyle(color: Colors.white), children: <TextSpan>[ TextSpan( text: 'Photo by ' ), TextSpan( style: TextStyle(fontWeight: FontWeight.bold), text: 'Chris Godley', ), ], ), ), ), ), ], ) , Padding( padding: const EdgeInsets.all(8.0), child: Column( crossAxisAlignment: CrossAxisAlignment.stretch, children: <Widget>[ Text('Artisans of Southern India', style: Theme.of(context).textTheme.bodyLarge), Text('Silk Spinners', style: Theme.of(context).textTheme.bodyMedium), Text('Sivaganga, Tamil Nadu', style: Theme.of(context).textTheme.bodySmall), ], ), ), ], ), ), ); } } class ItemGalleryBox extends StatelessWidget { const ItemGalleryBox(this.index, {super.key}); final int index; @override Widget build(BuildContext context) { final List<String> tabNames = <String>[ 'A', 'B', 'C', 'D', ]; return SizedBox( height: 200.0, child: DefaultTabController( length: tabNames.length, child: Column( children: <Widget>[ Expanded( child: TabBarView( children: tabNames.map<Widget>((String tabName) { return Container( key: PageStorageKey<String>(tabName), child: Padding( padding: const EdgeInsets.all(8.0), child: Card( child: Column( children: <Widget>[ Expanded( child: ColoredBox( color: Theme.of(context).primaryColor, child: Center( child: Text(tabName, style: Theme.of(context).textTheme.headlineSmall!.copyWith(color: Colors.white)), ), ), ), Row( children: <Widget>[ IconButton( icon: const Icon(Icons.share), onPressed: () { print('Pressed share'); }, ), IconButton( icon: const Icon(Icons.event), onPressed: () { print('Pressed event'); }, ), Expanded( child: Padding( padding: const EdgeInsets.only(left: 8.0), child: Text('This is item $tabName'), ), ), ], ), ], ), ), ), ); }).toList(), ), ), const TabPageSelector(), ], ), ), ); } } class BottomBar extends StatelessWidget { const BottomBar({super.key}); @override Widget build(BuildContext context) { return Container( decoration: BoxDecoration( border: Border( top: BorderSide( color: Theme.of(context).dividerColor, ), ), ), child: const Row( mainAxisAlignment: MainAxisAlignment.spaceBetween, children: <Widget>[ BottomBarButton(Icons.new_releases, 'News'), BottomBarButton(Icons.people, 'Requests'), BottomBarButton(Icons.chat, 'Messenger'), BottomBarButton(Icons.bookmark, 'Bookmark'), BottomBarButton(Icons.alarm, 'Alarm'), ], ), ); } } class BottomBarButton extends StatelessWidget { const BottomBarButton(this.icon, this.title, {super.key}); final IconData icon; final String title; @override Widget build(BuildContext context) { return Padding( padding: const EdgeInsets.all(8.0), child: Column( children: <Widget>[ IconButton( icon: Icon(icon), onPressed: () { print('Pressed: $title'); }, ), Text(title, style: Theme.of(context).textTheme.bodySmall), ], ), ); } } class GalleryDrawer extends StatelessWidget { const GalleryDrawer({ super.key }); void _changeTheme(BuildContext context, bool value) { ComplexLayoutApp.of(context)?.lightTheme = value; } void _changeScrollMode(BuildContext context, ScrollMode mode) { ComplexLayoutApp.of(context)?.scrollMode = mode; } @override Widget build(BuildContext context) { final ScrollMode currentMode = ComplexLayoutApp.of(context)!.scrollMode; return Drawer( // Note: for real apps, see the Gallery material Drawer demo. More // typically, a drawer would have a fixed header with a scrolling body // below it. child: ListView( key: const PageStorageKey<String>('gallery-drawer'), padding: EdgeInsets.zero, children: <Widget>[ const FancyDrawerHeader(), ListTile( key: const Key('scroll-switcher'), title: const Text('Scroll Mode'), onTap: () { _changeScrollMode(context, currentMode == ScrollMode.complex ? ScrollMode.tile : ScrollMode.complex); Navigator.pop(context); }, trailing: Text(currentMode == ScrollMode.complex ? 'Tile' : 'Complex'), ), ListTile( leading: const Icon(Icons.brightness_5), title: const Text('Light'), onTap: () { _changeTheme(context, true); }, selected: ComplexLayoutApp.of(context)!.lightTheme, trailing: Radio<bool>( value: true, groupValue: ComplexLayoutApp.of(context)!.lightTheme, onChanged: (bool? value) { _changeTheme(context, value!); }, ), ), ListTile( leading: const Icon(Icons.brightness_7), title: const Text('Dark'), onTap: () { _changeTheme(context, false); }, selected: !ComplexLayoutApp.of(context)!.lightTheme, trailing: Radio<bool>( value: false, groupValue: ComplexLayoutApp.of(context)!.lightTheme, onChanged: (bool? value) { _changeTheme(context, value!); }, ), ), const Divider(), ListTile( leading: const Icon(Icons.hourglass_empty), title: const Text('Animate Slowly'), selected: timeDilation != 1.0, onTap: () { ComplexLayoutApp.of(context)!.toggleAnimationSpeed(); }, trailing: Checkbox( value: timeDilation != 1.0, onChanged: (bool? value) { ComplexLayoutApp.of(context)!.toggleAnimationSpeed(); }, ), ), ], ), ); } } class FancyDrawerHeader extends StatelessWidget { const FancyDrawerHeader({super.key}); @override Widget build(BuildContext context) { return Container( color: Colors.purple, height: 200.0, child: const SafeArea( bottom: false, child: Placeholder(), ), ); } }