개요
이번 블로그 포스트에서는 Flutter에서 화면의 레이아웃을 담당하는 위젯들을 자세히 살펴보도록 하겠습니다.
이 블로그 포스트에서 소개하는 소스 코드는 아래에 링크에서 확인할 수 있습니다.
SafeArea
최신 스마트폰은 앱 화면을 최대한 활용할 수 있도록 디자인되었습니다. iOS의 노치(Notch) 디자인이나 안드로이드의 상태바 하단 등 앱이 표시되는 영역이 확대되었습니다.
그로 인해서, 다음과 같이 코드를 작성하면,
@override
Widget build(BuildContext context) {
return Scaffold(
body: Text('Hello world'),
);
}
다음과 같이 상태바 하단에 표시가 되는 문제들이 발생합니다.
data:image/s3,"s3://crabby-images/bef4d/bef4dc7eff7c2d1477a75803b712a5ddc4405e86" alt="Flutter - Before SafeArea widget"
이런 문제를 해결하기 위해, 사용할 수 있는 위젯이 SafeArea
입니다. SafeArea
는 다음과 같이 사용할 수 있습니다.
@override
Widget build(BuildContext context) {
return Scaffold(
body: SafeArea(
child: Text('Hello world'),
),
);
}
이렇게 SafeArea
를 사용하면 다음과 같이 상태바와 중복되지 않게 위젯들을 표시할 수 있습니다.
data:image/s3,"s3://crabby-images/453fe/453fef4ac2d79324c8b94159d043d5bdf64d9a9b" alt="Flutter - After SafeArea widget"
Center
Center
위젯은 자식 위젯을 중앙에 표시합니다.
@override
Widget build(BuildContext context) {
return Scaffold(
body: Center(
child: Text('Hello world'),
),
);
}
이렇게 Center 위젯의 자식 위젯으로 Text
을 전달하면, 다음과 같이 중앙에 표시되는 것을 확인할 수 있습니다.
data:image/s3,"s3://crabby-images/c0391/c0391f29ad5ea893a4fac3e5d5c45e0a1d4c5e3d" alt="Flutter - Center widget"
Padding
Padding
위젯는 자식 위젯 주위에 padding
을 표시합니다.
@override
Widget build(BuildContext context) {
return Scaffold(
body: Padding(
child: Text('Hello world'),
padding: EdgeInsets.fromLTRB(100.0, 300.0, 10.0, 40.0),
),
);
}
이렇게 Padding 위젯에 자식 위젯을 지정하고, 원하는 padding
값을 설정하면 다음과 같이 자식 위젯가 Padding과 함께 표시됩니다.
data:image/s3,"s3://crabby-images/5a48c/5a48c8381c8a9437c9ee9941809b104426ad37b0" alt="Flutter - Padding widget"
Container
Container
위젯을 사용하면, 기본적으로 최대한의 공간을 차지하여 표시됩니다.
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text('Container'),
),
body: Container(
color: Colors.red,
),
);
}
이렇게 Container를 사용하면 다음과 같이 최대한의 공간(화면 전체)에 표시됩니다.
data:image/s3,"s3://crabby-images/e2e78/e2e780d23ac9ccf4a5a3a28b5e588180aa272ba1" alt="Flutter - Container widget with no option"
만약, Container에 자식 위젯이 제공되면, Container는 자식 위젯의 사이즈로 표시됩니다.
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text('Container'),
),
body: Container(
color: Colors.red,
child: Text(
'Hello world',
style: TextStyle(color: Colors.yellow),
),
),
);
}
이렇게 Container에 자식 위젯을 지정하면 다음과 같이 Container는 자식 위젯의 사이즈로 표시됩니다.
data:image/s3,"s3://crabby-images/a6a08/a6a08f1023c047bb6d221c99b35932fb37474130" alt="Flutter - Container widget with child"
Container는 Padding
과는 다르게 다음과 같이 여러 파라메터들을 사용할 수 있습니다.
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text('Container'),
),
body: Container(
child: Text(
'Hello world',
style: TextStyle(color: Colors.red),
),
padding: EdgeInsets.fromLTRB(20.0, 30.0, 40.0, 50.0),
margin: EdgeInsets.all(100.0),
width: 200,
height: 100,
transform: Matrix4.rotationZ(0.5),
decoration: BoxDecoration(
color: Colors.yellow,
border: Border.all(color: Colors.black, width: 3),
borderRadius: BorderRadius.all(Radius.circular(18)),
boxShadow: const [
BoxShadow(blurRadius: 10),
],
),
),
);
}
이렇게 여러 파라메터들을 사용하면 다음과 같이 파라메터가 적용된 Container 위젯이 표시됩니다.
data:image/s3,"s3://crabby-images/04a37/04a370519fa63e58068362c3c72a5aa15599617b" alt="Flutter - Container widget with options"
Column과 Row
Flutter에서 화면 레이아웃에 가장 많이 사용되는 Column
위젯과 Row
위젯에 대해서 살펴보도록 하겠습니다.
Column
Column
위젯은 자식 위젯들을 세로로 정렬하여 표시합니다. Column 위젯은 세로의 모든 영역을 차지하게 됩니다.
@override
Widget build(BuildContext context) {
return Scaffold(
body: Column(
children: [
Container(
width: 100,
height: 100,
color: Colors.red,
),
Container(
width: 100,
height: 100,
color: Colors.yellow,
),
Container(
width: 100,
height: 100,
color: Colors.green,
),
],
),
);
}
위와 같이 Column 위젯은 children
으로 전달 받은 위젯들을 다음과 같이 세로로 표시합니다.
data:image/s3,"s3://crabby-images/ee4a6/ee4a66c5a44851e1ae14ce013a99c23ab4afe343" alt="Flutter - Column widget"
Column의 mainAxisAlignment
파라메터를 이용하면, 자식 위젯들의 정렬을 변경할 수 있습니다.
- center
@override
Widget build(BuildContext context) {
return Scaffold(
body: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Container(
width: 100,
height: 100,
color: Colors.red,
),
Container(
width: 100,
height: 100,
color: Colors.yellow,
),
Container(
width: 100,
height: 100,
color: Colors.green,
),
],
),
);
}
data:image/s3,"s3://crabby-images/b416b/b416bcf016e6d44d3b93d030ccaee4080e3b4edd" alt="Flutter - Column widget center"
- end
@override
Widget build(BuildContext context) {
return Scaffold(
body: Column(
mainAxisAlignment: MainAxisAlignment.end,
children: [
Container(
width: 100,
height: 100,
color: Colors.red,
),
Container(
width: 100,
height: 100,
color: Colors.yellow,
),
Container(
width: 100,
height: 100,
color: Colors.green,
),
],
),
);
}
data:image/s3,"s3://crabby-images/41598/41598a7b127a08ef982b641cd60a4e59c326cc60" alt="Flutter - Column widget end"
- spaceAround
@override
Widget build(BuildContext context) {
return Scaffold(
body: Column(
mainAxisAlignment: MainAxisAlignment.spaceAround,
children: [
Container(
width: 100,
height: 100,
color: Colors.red,
),
Container(
width: 100,
height: 100,
color: Colors.yellow,
),
Container(
width: 100,
height: 100,
color: Colors.green,
),
],
),
);
}
data:image/s3,"s3://crabby-images/0b5ff/0b5ffb435029a5a26c67a8954612d43f87a9d5a5" alt="Flutter - Column widget spaceAround"
Row
Row
위젯은 자식 위젯들을 가로로 정렬하여 표시합니다. Row 위젯은 가로의 모든 영역을 차지하게 됩니다.
@override
Widget build(BuildContext context) {
return Scaffold(
body: Row(
children: [
Container(
width: 100,
height: 100,
color: Colors.red,
),
Container(
width: 100,
height: 100,
color: Colors.yellow,
),
Container(
width: 100,
height: 100,
color: Colors.green,
),
],
),
);
}
위와 같이 Row 위젯은 children
으로 전달 받은 위젯들을 다음과 같이 가로로 표시합니다.
data:image/s3,"s3://crabby-images/2ce75/2ce756838a729b800429d3976693ee8503be92c8" alt="Flutter - Row widget"
Row의 mainAxisAlignment
파라메터를 이용하면, 자식 위젯들의 정렬을 변경할 수 있습니다.
- center
@override
Widget build(BuildContext context) {
return Scaffold(
body: Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Container(
width: 100,
height: 100,
color: Colors.red,
),
Container(
width: 100,
height: 100,
color: Colors.yellow,
),
Container(
width: 100,
height: 100,
color: Colors.green,
),
],
),
);
}
data:image/s3,"s3://crabby-images/722ad/722ade98d768e4dda15eb5dfe6ba9d34d03a285e" alt="Flutter - Row widget center"
- end
@override
Widget build(BuildContext context) {
return Scaffold(
body: Row(
mainAxisAlignment: MainAxisAlignment.end,
children: [
Container(
width: 100,
height: 100,
color: Colors.red,
),
Container(
width: 100,
height: 100,
color: Colors.yellow,
),
Container(
width: 100,
height: 100,
color: Colors.green,
),
],
),
);
}
data:image/s3,"s3://crabby-images/cb622/cb6226dae8acc40448beb116d1ae6bcdb0ae0bd3" alt="Flutter - Row widget end"
- spaceAround
@override
Widget build(BuildContext context) {
return Scaffold(
body: Row(
mainAxisAlignment: MainAxisAlignment.spaceAround,
children: [
Container(
width: 100,
height: 100,
color: Colors.red,
),
Container(
width: 100,
height: 100,
color: Colors.yellow,
),
Container(
width: 100,
height: 100,
color: Colors.green,
),
],
),
);
}
data:image/s3,"s3://crabby-images/2a2a7/2a2a7acab9ffd21f476bd3c879181c0662f53caa" alt="Flutter - Row widget spaceAround"
Expanded
Expanded
위젯은 Column
위젯 또는 Row
위젯과 함께 사용하여 웹 개발에서 사용되는 flex
와 같은 기능을 구현할 수 있습니다.
@override
Widget build(BuildContext context) {
return Scaffold(
body: Column(
children: [
Expanded(
child: Container(
color: Colors.red,
),
flex: 3,
),
Expanded(
child: Container(
color: Colors.yellow,
),
flex: 1,
),
Expanded(
child: Container(
color: Colors.green,
),
flex: 2,
),
],
),
);
}
위와 같이 Column 위젯의 자식 위젯으로 Expanded 위젯을 사용하고, Expanded 위젯의 flex
파라메터를 사용하면 다음과 같이 웹에서 사용되는 flex
와 같은 기능을 구현할 수 있습니다.
data:image/s3,"s3://crabby-images/8f23c/8f23cb7ec5c924501e02859e43340ba001c1c5be" alt="Flutter - Expanded widget"
Stack
Stack
위젯을 사용하면, 위젯 위에 위젯을 표시할 수 있습니다.
@override
Widget build(BuildContext context) {
return Scaffold(
body: Stack(
children: [
Container(
width: 400,
height: 400,
color: Colors.green,
),
Container(
width: 200,
height: 200,
color: Colors.yellow,
),
Container(
width: 50,
height: 50,
color: Colors.red,
),
],
),
);
}
위와 같이 Stack 위젯을 사용하면, 다음과 같이 위젯 위에 위젯이 표시되는 것을 확인할 수 있습니다.
data:image/s3,"s3://crabby-images/d4750/d4750460358e24e55bcb8b59ae0a6bec43e68c44" alt="Flutter - Stack widget"
SizedBox
SizedBox
위젯은 위젯과 위젯 사이에 빈 공간을 추가할 때, 자주 사용됩니다.
@override
Widget build(BuildContext context) {
return Scaffold(
body: Column(
children: [
Container(
color: Colors.red,
width: 100,
height: 100,
),
SizedBox(
height: 200,
),
Container(
color: Colors.yellow,
width: 100,
height: 100,
),
SizedBox(
height: 50,
),
Container(
color: Colors.green,
width: 100,
height: 100,
),
],
),
);
}
위와 같이 Container 위젯과 위젯 사이에 SizedBox 위젯을 추가하면 다음과 같이 빈 공간이 잘 추가되는 것을 확인할 수 있습니다.
data:image/s3,"s3://crabby-images/51d36/51d36d59f671a7a4e047ff27585ddc74fcd6e5de" alt="Flutter - SizedBox widget"
완료
이것으로 Flutter에서 자주 사용되는 레이아웃 관련 위젯들을 살펴보았습니다. 이제 이 위젯들을 사용하여 다양하게 레이아웃을 구성해 보시기 바랍니다.
제 블로그가 도움이 되셨나요? 하단의 댓글을 달아주시면 저에게 큰 힘이 됩니다!
앱 홍보
Deku
가 개발한 앱을 한번 사용해보세요.Deku
가 개발한 앱은 Flutter로 개발되었습니다.관심있으신 분들은 앱을 다운로드하여 사용해 주시면 정말 감사하겠습니다.