data:image/s3,"s3://crabby-images/345ac/345ac975fcf02a7b855b2667cb04670a6fedc9bb" alt="컬럼 연습하기"
1. 박스 1개 만들기
data:image/s3,"s3://crabby-images/4ecc3/4ecc30b6ca54ea59fdc1fff52f376650c5f01804" alt="notion image"
- double.infinity는 무한대
data:image/s3,"s3://crabby-images/b38f1/b38f1741ebbc2fb7b81df4586251e720fcf2f683" alt="notion image"
2. 박스 3개 만들기
- 컬럼은 세로 정렬
- 로우는 가로 정렬
import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; import 'package:recipe_app2/components/recipe_menu_item.dart'; import '../components/recipe_body.dart'; import '../components/recipe_menu.dart'; import '../components/recipe_title.dart'; class RecipePage extends StatelessWidget { const RecipePage({super.key}); @override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.blue, width: 50, height: 50, ), Container( color: Colors.yellow, width: 50, height: 50, ) ], ), // 새로 방향 레이아웃 -> 자식이 없을때 크기 : 0, 0, 0, 0 ); } AppBar _appbar() { return AppBar( actions: [ Icon(CupertinoIcons.search), //시그니처 매개변수 키 값이 없다, 그 위젯의 핵심은 시그니처로 키 값을 넣지 않는다. SizedBox(width: 15), Icon( CupertinoIcons.heart, color: Colors.redAccent, ), SizedBox(width: 15), ], ); } }
data:image/s3,"s3://crabby-images/86ae7/86ae7d1426a863fba1c9e5a5e6163e932fd8e1fd" alt="notion image"
- 컬럼 // 세로 : 블락, 가로 : 인라인
- 로우 // 가로 : 블락, 세로 : 인라인
- 자식이 큰게 들어오면 다 깨짐 → 컬럼 자체를 미리 키워놔야 함
data:image/s3,"s3://crabby-images/138ab/138ab0d2ad2eba7e34887849ca64fcee9ede78df" alt="notion image"
- 컨테이너는 끝까지 차지함
- 자식이 없으면 블락
- 자식이 있으면 인라인
data:image/s3,"s3://crabby-images/4ee2b/4ee2b43d1698e5e3aef5669aa5f70a659a2a7f3c" alt="notion image"
data:image/s3,"s3://crabby-images/63ef4/63ef4fb30ecc9e3a2a16b716d22dcfe4311f57a5" alt="notion image"
- ListView는 최소 값이 가로가 double.infinity라서 width가 안보임
크기를 조절하려면 부모를 만들어서 크기 조절하기
data:image/s3,"s3://crabby-images/8efc5/8efc5040d3d045cdec355f0c75f92232874338ff" alt="notion image"
3. 외워야함!!
ㅤ | 인라인 | 블락 | 스타트 | 센터 |
Colum | 가로 | 세로 | 세로 | 가로 |
Row | 세로 | 가로 | 가로 | 세로 |
Container | 자식O | 자식 X | ㅤ | ㅤ |
ListView | 세로 | 세로 | ㅤ | ㅤ |
4. 서로에게 영향 안받도록 배치하기
- 로우는 가로 정렬
crossAxisAlignment: CrossAxisAlignment.start // 왼쪽에서 시작
import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; import '../components/recipe_title.dart'; class RecipePage extends StatelessWidget { const RecipePage({super.key}); @override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.blue, width: 50, height: 50, ), Row( children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.blue, width: 50, height: 50, ), ], ), ], ), ); } AppBar _appbar() { return AppBar( actions: [ Icon(CupertinoIcons.search), SizedBox(width: 15), Icon( CupertinoIcons.heart, color: Colors.redAccent, ), SizedBox(width: 15), ], ); } }
data:image/s3,"s3://crabby-images/1f58d/1f58d585a3bcc449a85d2d50e020dd4d884abf7e" alt="notion image"
5. 각각 독립된 영역으로 구분하기
- 서로 영향을 주고 받지 않도록 해야 디자인이 깨지지 않음
import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; import '../components/recipe_title.dart'; class RecipePage extends StatelessWidget { const RecipePage({super.key}); @override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.blue, width: 50, height: 50, ), Row( children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.blue, width: 50, height: 50, ), ], ), Row( mainAxisAlignment: MainAxisAlignment.center, children: [ Container( color: Colors.yellow, width: 50, height: 50, ), ], ), ], ), ); } AppBar _appbar() { return AppBar( actions: [ Icon(CupertinoIcons.search), SizedBox(width: 15), Icon( CupertinoIcons.heart, color: Colors.redAccent, ), SizedBox(width: 15), ], ); } }
data:image/s3,"s3://crabby-images/eb4c3/eb4c3de0594b70975380844bb5afc5b1b824c893" alt="notion image"
- 영역이 나뉘어져 있음
- 각 영역마다 3개의 컴포넌트로 나눠야 함
- 컴포넌트끼리 다 독립적이지 않아서 영향을 받는 것 → 완전히 독립시켜야 함
- 전체는 컬럼 : 위에서 아래로
내부에 컴포넌트 각각 3개로 구분
data:image/s3,"s3://crabby-images/d2c61/d2c6132c7e126c064c48e8cafe3f4b87c9b1cd03" alt="notion image"
6. 디자인이 안 무너지게 설계하기
- 컴포넌트로 뺀 후 타입을 변경한 경우 부모의 타입으로 바꿔줘야 오류가 나지 않음
import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; import '../components/recipe_title.dart'; class RecipePage extends StatelessWidget { const RecipePage({super.key}); @override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( // 전체 디자인 // 총 3개의 섹션 children: [ _top(), Row( children: [ Container( color: Colors.red, width: 50, ) ], ) ], ), ); } Widget _top() { // 부모 타입으로 변경하기 return Container( width: double.infinity, child: Column( children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.yellow, width: 50, height: 50, ) ], ), ); } AppBar _appbar() { return AppBar( actions: [ Icon(CupertinoIcons.search), SizedBox(width: 15), Icon( CupertinoIcons.heart, color: Colors.redAccent, ), SizedBox(width: 15), ], ); } }
data:image/s3,"s3://crabby-images/ed186/ed186e7d6261d9371cb8e58562516d3a03c9e497" alt="notion image"
- top 디자인하기
import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; import '../components/recipe_title.dart'; class RecipePage extends StatelessWidget { const RecipePage({super.key}); @override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( // 전체 디자인 // 총 3개의 섹션 children: [ _top(), Row( children: [ Container( color: Colors.red, width: 50, ) ], ) ], ), ); } Widget _top() { // 부모 타입으로 변경하기 return Container( width: double.infinity, child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.yellow, width: 50, height: 50, ) ], ), ); } AppBar _appbar() { return AppBar( actions: [ Icon(CupertinoIcons.search), SizedBox(width: 15), Icon( CupertinoIcons.heart, color: Colors.redAccent, ), SizedBox(width: 15), ], ); } }
data:image/s3,"s3://crabby-images/459bd/459bd53c294e1f88718931d0039d3ff5a2dc5b94" alt="notion image"
data:image/s3,"s3://crabby-images/c31d9/c31d9f159a1817c58b934303480fadc33c05882c" alt="notion image"
data:image/s3,"s3://crabby-images/4e737/4e7377ee2a5c525881b5421a692ae61f4691c044" alt="notion image"
- 하나만 길이가 다른 경우
@override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( // 전체 디자인 // 총 3개의 섹션 children: [ _top(), Row( children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.yellow, width: 50, height: 100, ) ], ) ], ), ); } Widget _top() { // 부모 타입으로 변경하기 return Container( width: double.infinity, child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.yellow, width: 50, height: 50, ) ], ), ); }
data:image/s3,"s3://crabby-images/82617/826178b3c5542a289e96a078fa7bb7600497aad7" alt="notion image"
data:image/s3,"s3://crabby-images/8a632/8a632962c59975ad30537aea5a818962f9451600" alt="notion image"
@override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( // 전체 디자인 // 총 3개의 섹션 children: [ _top(), Container( height: 100, // 자식의 크기에 맞추려면 자식 각각에 높이를 주면 됨 child: Row( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( color: Colors.red, width: 50, height: double.infinity, ), Container( color: Colors.green, width: 50, height: double.infinity, ), Container( color: Colors.yellow, width: 50, height: double.infinity, ) ], ), ) ], ), );
- 자식의 높이는 infinity로 맞추고 부모의 길이 조절하기
data:image/s3,"s3://crabby-images/ddf54/ddf54f3429d081c523b2e5f9fddb191493246032" alt="notion image"
data:image/s3,"s3://crabby-images/b9e03/b9e03620a1bb433da80a3a5fcd3a61fcaa3039fd" alt="notion image"
data:image/s3,"s3://crabby-images/62bac/62bac02a34a5de95274f8b78099076d358b9b545" alt="notion image"
@override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( // 전체 디자인 // 총 3개의 섹션 children: [ _top(), _center(), Container( width: 50, height: 50, child: Container( color: Colors.red, ), ) ], ), );
data:image/s3,"s3://crabby-images/c27bf/c27bfcd7acf77ebd5b5a36605e8b154ca79f8f27" alt="notion image"
@override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( // 전체 디자인 // 총 3개의 섹션 children: [ _top(), _center(), Center( child: Container( width: 50, height: 50, child: Container( color: Colors.red, ), ), ), ], ) ); }
data:image/s3,"s3://crabby-images/bba09/bba0933302aa54948f22ea878c58bef3da2e1b33" alt="notion image"
@override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( // 전체 디자인 // 총 3개의 섹션 children: [ _top(), _center(), Container( width: double.infinity, height: 50, child: Row( mainAxisAlignment: MainAxisAlignment.center, children: [ Container( color: Colors.red, width: 50, height: 50, ), ], ), ), ], ) ); }
data:image/s3,"s3://crabby-images/19f16/19f16af5f0a7094ab0281aee3a32e72712997995" alt="notion image"
무너지지 않는 디자인 만들기
- 앱은 가로 사이즈가 제일 중요함
- 세로 크기는 안에 있는 자식의 크기에 맞춘 것이기에 신경쓸 필요 없음
Widget _top() { // 부모 타입으로 변경하기 return SizedBox( width: double.infinity, child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.yellow, width: 50, height: 50, ) ], ), ); }
- top은 colum → 가로 : 인라인 → 늘려주기
높이 : 자식의 크기에 맞추기
- 박스로 감싸는 이유는 가로 설계 때문 → 로우는 그대로 두기
data:image/s3,"s3://crabby-images/2d12c/2d12c04999307d7f2dbb1cbdb3949a6fb34027af" alt="notion image"
data:image/s3,"s3://crabby-images/aa051/aa0512619e62850cbb99cc448cc4424db585257d" alt="notion image"
@override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( crossAxisAlignment: CrossAxisAlignment.center, // 전체 디자인 // 총 3개의 섹션 children: [ _top(), // 새로로 내려감 _center(), Align( alignment: Alignment.centerRight, child: Container( color: Colors.red, width: 50, height: 50, ), ), ], ) ); }
data:image/s3,"s3://crabby-images/5eb77/5eb77aa850675f87b7ae9268755ae6e624ed188a" alt="notion image"
import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; import 'package:flutter/widgets.dart'; import '../components/recipe_title.dart'; class RecipePage extends StatelessWidget { const RecipePage({super.key}); @override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( // 전체 디자인 // 총 3개의 섹션 children: [ _top(), _center(), _bottom(), ], ) ); } Widget _bottom() { return Align( alignment: Alignment.center, child: Container( color: Colors.red, width: 50, height: 50, ), ); } Widget _center() { return Row( // 부모로 감싸는 이유는 가로 설계 때문 crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.yellow, width: 50, height: 50, ) ], ); } Widget _top() { // 부모 타입으로 변경하기 return SizedBox( width: double.infinity, child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.yellow, width: 50, height: 50, ) ], ), ); } AppBar _appbar() { return AppBar( actions: [ Icon(CupertinoIcons.search), SizedBox(width: 15), Icon( CupertinoIcons.heart, color: Colors.redAccent, ), SizedBox(width: 15), ], ); } }
data:image/s3,"s3://crabby-images/ba8e3/ba8e3d733d6861c2706d091e71108b54b7d65065" alt="notion image"
data:image/s3,"s3://crabby-images/dd41b/dd41ba96896702e1d0d17232c2ee5ac7aed9fedc" alt="notion image"
[ 노란색을 가운데 두고 싶다 ]
data:image/s3,"s3://crabby-images/9c262/9c2625ea5ef2b333481926f30e934b401b78b029" alt="notion image"
다시 Row로 감싼다 (컨테이너에 alt + enter)
data:image/s3,"s3://crabby-images/6b5f6/6b5f6d930f212cd844585a90eb3cf450f9229ce1" alt="notion image"
data:image/s3,"s3://crabby-images/f4aef/f4aefdc2652e5d77f89e72f458841aa9b3c675db" alt="notion image"
근데 이렇게 하면 안 됨
why?
data:image/s3,"s3://crabby-images/b7d0a/b7d0a257e69e922fe9b34acc30841a4ea23a1753" alt="notion image"
배치할 때, 독립적이지 않으니까 모두 영향을 받는다. 독립적으로 디자인해야 함! 지금 컬럼 안에 다 때려넣었잖아. 전체도 컬럼, 안에 세부(?)도 컬럼. 이 3개가 컬럼 안에 묶여야지 컴포넌트화 시킬 수 있음
[ 전체 코드 ] - 다음에 연습
import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; import 'package:recipe_app/components/recipe_body.dart'; import 'package:recipe_app/components/recipe_menu.dart'; import 'package:recipe_app/components/recipe_menu_item.dart'; import '../components/recipe_title.dart'; class RecipePage extends StatelessWidget { const RecipePage({super.key}); @override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.blue, width: 50, height: 50, ), Row( children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.blue, width: 50, height: 50, ), ], ), Row( mainAxisAlignment: MainAxisAlignment.center, children: [ Container( color: Colors.yellow, width: 50, height: 50, ), ], ), ], ), ); } AppBar _appbar() { return AppBar( actions: [ Icon(CupertinoIcons.search), SizedBox(width: 15), Icon( CupertinoIcons.heart, color: Colors.redAccent, ), SizedBox(width: 15), ], ); } }
이건 컴포넌트화 불가능. 독립되어 있지 않음.
따로 박스 설계를 하지 않으면 디자인이 무너질 것
[ 제대로 설계 Column ]
data:image/s3,"s3://crabby-images/e42e9/e42e955570a0c38aa94d1d5f3d60a7c513a2008f" alt="notion image"
섹션이 총 3개!
data:image/s3,"s3://crabby-images/5fa9d/5fa9d0bf740ae2afff3ca211c60aca8bd24bc07f" alt="notion image"
이런 식으로 디자인이 나와야 함
data:image/s3,"s3://crabby-images/d8c08/d8c08241746b8c479618c6d66bf5604916539e78" alt="notion image"
다 만들었으니 애를 컴포넌트로 빼자 (컴포넌트로 빼는게 맞는데 지금은 수업한다고 Method로 뺌)
data:image/s3,"s3://crabby-images/cc628/cc628d147117959dfd16cf18be25ba3784afc293" alt="notion image"
data:image/s3,"s3://crabby-images/bc2cb/bc2cbcbce8bbbe79edfd296bd33e869cdc0a99fb" alt="notion image"
컨테이너로 감싸고, 리턴 타입을 Widget (부모)로 바꿔준다.
data:image/s3,"s3://crabby-images/a4443/a4443c096a56f5dd248501e5f6a75fb3de6a7758" alt="notion image"
data:image/s3,"s3://crabby-images/91970/919706854081c74c2e309fa27c5b8a54cc3b47b2" alt="notion image"
독립됨
import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; import 'package:recipe_app/components/recipe_body.dart'; import 'package:recipe_app/components/recipe_menu.dart'; import 'package:recipe_app/components/recipe_menu_item.dart'; import '../components/recipe_title.dart'; class RecipePage extends StatelessWidget { const RecipePage({super.key}); @override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( children: [ _top(), Row(), ], ), ); } Widget _top() { return Container( width: double.infinity, child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.yellow, width: 50, height: 50, ), ], ), ); } AppBar _appbar() { return AppBar( actions: [ Icon(CupertinoIcons.search), SizedBox(width: 15), Icon( CupertinoIcons.heart, color: Colors.redAccent, ), SizedBox(width: 15), ], ); } }
Row
data:image/s3,"s3://crabby-images/26d34/26d3434e13be5c46dae8a2bda5e007c1030c6ae1" alt="notion image"
import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; class RecipePage extends StatelessWidget { const RecipePage({super.key}); @override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( children: [ _top(), Container( height: 200, child: Row( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( color: Colors.red, width: 50, height: double.infinity, ), Container( color: Colors.green, width: 50, height: double.infinity, ), Container( color: Colors.yellow, width: 50, height: double.infinity, ), ], ), ), ], ), ); } Widget _top() { return Container( width: double.infinity, child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.yellow, width: 50, height: 50, ), ], ), ); } AppBar _appbar() { return AppBar( actions: [ Icon(CupertinoIcons.search), SizedBox(width: 15), Icon( CupertinoIcons.heart, color: Colors.redAccent, ), SizedBox(width: 15), ], ); } }
어휴 ssi bv
import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; class RecipePage extends StatelessWidget { const RecipePage({super.key}); @override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( children: [ _top(), _center(), Container( width: double.infinity, height: 50, child: Container( color: Colors.red, width: 50, height: double.infinity, ), ), ], ), ); } Container _center() { return Container( height: 200, child: Row( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( color: Colors.red, width: 50, height: double.infinity, ), Container( color: Colors.green, width: 50, height: double.infinity, ), Container( color: Colors.yellow, width: 50, height: double.infinity, ), ], ), ); } Widget _top() { return Container( width: double.infinity, child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.yellow, width: 50, height: 50, ), ], ), ); } AppBar _appbar() { return AppBar( actions: [ Icon(CupertinoIcons.search), SizedBox(width: 15), Icon( CupertinoIcons.heart, color: Colors.redAccent, ), SizedBox(width: 15), ], ); } }
data:image/s3,"s3://crabby-images/15a8c/15a8ca971e899b91e2dcd9cb9bb5befc8d603046" alt="notion image"
data:image/s3,"s3://crabby-images/d1b68/d1b68f9362fdfcc2467c08fb48a1dacc06343457" alt="notion image"
height는 박스로 감싸서 크기 줄 필요 없음
Column은 늘리고, Row는 그대로 둔다.
Column은 왜 늘릴까? 처음에 Row가 들어갈 수 있는 경우도 있겠지
data:image/s3,"s3://crabby-images/b880c/b880ce8a9c8e7b664d98eba13f1abd48b066cc36" alt="notion image"
center를 두든, 뭘 두든…
[ Align ]
data:image/s3,"s3://crabby-images/8fb9e/8fb9e1b1b89456c13d847fbe4ebee797d2faf15c" alt="notion image"
data:image/s3,"s3://crabby-images/e859c/e859c9a6044f24719b9ae8746f55d333c7e42af0" alt="notion image"
data:image/s3,"s3://crabby-images/e97e2/e97e2ff77973f1c2e5e557a4740beb7d3e8075b8" alt="notion image"
끝까지 차지하기 위해선 컬럼은 SizedBox 사용
return 타입은 전부 위젯
import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; import 'package:flutter/widgets.dart'; class RecipePage extends StatelessWidget { const RecipePage({super.key}); @override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Column( children: [ _top(), _center(), _bottom(), ], ), ); } Align _bottom() { return Align( alignment: Alignment.centerRight, child: Container( color: Colors.red, width: 50, height: 50, ), ); } Widget _center() { return Row( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.yellow, width: 50, height: 50, ), ], ); } Widget _top() { return SizedBox( width: double.infinity, child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( color: Colors.red, width: 50, height: 50, ), Container( color: Colors.green, width: 50, height: 50, ), Container( color: Colors.yellow, width: 50, height: 80, ), ], ), ); } AppBar _appbar() { return AppBar( actions: [ Icon(CupertinoIcons.search), SizedBox(width: 15), Icon( CupertinoIcons.heart, color: Colors.redAccent, ), SizedBox(width: 15), ], ); } }
data:image/s3,"s3://crabby-images/78a68/78a687d910e421d92eb398f88f8ae33c5123ab19" alt="notion image"
배치는 Block을 만든다.
세로를 Block으로
- 컨테이너는 제약조건을 줄 수 있음
부모가 자식에게 제약 조건을 검 → 부모의 제약 조건을 알아야하기에 동적임
- 제약 조건을 모든 위젯이 들고 있는 것은 아님
sizesBox는 없음
import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; import 'package:flutter/widgets.dart'; import 'package:recipe_app/components/recipe_body.dart'; import 'package:recipe_app/components/recipe_menu.dart'; import 'package:recipe_app/components/recipe_menu_item.dart'; import '../components/recipe_title.dart'; class RecipePage extends StatelessWidget { const RecipePage({super.key}); @override Widget build(BuildContext context) { return Scaffold( appBar: _appbar(), body: Container( constraints: BoxConstraints( minWidth: 100, minHeight: 100, maxWidth: 150, maxHeight: 300, ), child: Container( alignment: Alignment.center, color: Colors.red, width: 1000, height: 1000, ), ), ); } AppBar _appbar() { return AppBar( actions: [ Icon(CupertinoIcons.search), SizedBox(width: 15), Icon( CupertinoIcons.heart, color: Colors.redAccent, ), SizedBox(width: 15), ], ); } }
Share article