「ウィジェットが思うような大きさにならない…」
本記事でこの悩みが解決できるかもしれません。
Flutter 公式のウィジェットの大きさ等に関するドキュメント、
Understanding Constraints を日本語で解説します。
ウィジェットの大きさの決定の仕方について、
仕組みから知りたい、というあなたにピッタリの記事となっています。
記事内では、様々な実装例をコードと横並びで確認できるWEBアプリを紹介します。
ぜひ読んでみてください!
Flutter のレイアウトの仕方
レイアウトのルール
Flutter のレイアウトは以下のルールに則って行われます。
Constraints go down. Sizes go up. Parent sets position.
日本語で解説すると、
制約は下ります。大きさは上ります。親が位置を決めます。
です。
より詳細に述べると以下になります。
- ウィジェットは親(parent)から自身の制約(constraints)を取得します。
constraintは4つのdouble型の値(最小と最大の幅(width),最小と最大の高さ(height))
の集合です。 - その後、ウィジェットはその制約を自身の子たち(children)に渡します。
ひとつひとつの子ウィジェットが、その子たちに自身の制約を伝えます。
(この制約はそれぞれの子で異なります。)
そして、どれくらいの大きさになりたいか確認します。 - ウィジェットは、子たちを配置します。
- 最終的にウィジェットは親に自身の大きさ(size)を伝えます。
(もちろん、制約の範囲内で伝えます。)
例
上で説明したルールでのレイアウト構築について、
図のようなレイアウト構築で行われるやりとりを、
ウィジェット(Widget)を擬人化し、会話する様子で紹介します。
お父さん(親Widget)、
私の制約はなんですか?
幅は、80~250 pixelの間で、
高さは30~85 pixelの間で設定して欲しいな。
なるほど…
自分としては5 pixel のpadding は欲しいから、
子どもたちには、
幅は240 pixel以下 (250 – 5 x 2)に、
高さは75 pixel 以下 (85 – 5 x 2)に
してもらうよう伝えればいいな。
子1くん、
あなたは、
幅は0 ~ 240 pixel の間で、
高さは0 ~ 75pixel の間で頼みます。
OK!
じゃあ幅は240 pixel、高さは20pixel がいいな!
ありがとう!
なるほど…
子2ちゃんは子1くんの下に配置したいから、
高さが残り55pixelだな。
子2ちゃん、
あなたは、
幅は0 ~ 240 pixel の間で、
高さは0 ~ 55 pixel の間で頼みます。
OK!
じゃあ幅は90 pixel、高さは30pixel がいいな!
ありがとう!
子どもたちの大きさが決まったぞ!
それじゃあ、子1くんは x : 5, y : 5 の位置に
子2ちゃんは x : 80, y : 25 の位置に配置しよう。
自分の大きさは幅が250pixel、
高さが60 pixel だな。
お父さん、
自分の大きさは、
幅が250pixel、高さが60 pixel でお願いします!
以上が具体例となります!
制限について
上でお伝えしたルールの結果として、
Flutterのレイアウトエンジンにはいくつか重要な制限があります。
- ウィジェットは自身の大きさを親から与えられた制約の中で決定できます。
これはつまり、ウィジェットはいつも欲しい大きさになれるわけではない、
ということです。 - ウィジェットは画面上での自身の位置を知ることはできないし、決定しません。
これは、そのウィジェットの親が位置を決めるからです。 - 親の大きさと位置は、さらにその親に依存するため、
ツリー全体を考慮せずにウィジェットの大きさと位置を正確に定義することは
不可能です。 - もしウィジェットがその親から異なる大きさを希望し、
親がウィジェットを配置するのに十分な情報を持っていないなら、
子の大きさは無視される可能性があります。
配置の情報を定義する時は、具体的にしてください。
レイアウトの具体例
“Understanding Constraints”では上記を踏まえたレイアウトの具体例として、
29の例が紹介されています。
これらの例について、確認できるWEBアプリを用意しました。
コードを横並びにして、
日本語にて例を確認できます。
ぜひこちらで様々なウィジェットの組み合わせによるレイアウトの
コードと結果を確認してみてください。
制約の Tight vs. Loose
制約がTightである、Looseである、という言葉の意味について解説します。
Tight
Tight(タイト)な制約は1つの大きさを指定します。
言い換えればTightな制約は幅の最大値と最小値が同じであり、
高さの最大値と最小値が同じである制約です。
Flutterのbox.dart
のBoxConstraints
コンストラクタでは以下のように定義されています。
BoxConstraints.tight(Size size)
: minWidth = size.width,
maxWidth = size.width,
minHeight = size.height,
maxHeight = size.height;
Example 2では赤いContainer
が強制的に表示領域を埋め尽くすようになりました。
これは表示領域がTightな制約をContainer
に与えていたからです。
Loose
一方でLoose(ルーズ)な制約は、最大の幅と高さが設定されていますが、
ウィジェットは好きなだけ小さくなることができます。
つまり、最小の幅と高さは0
です。
Flutterのbox.dart
のBoxConstraints
コンストラクタでは以下のように定義されています。
BoxConstraints.loose(Size size)
: minWidth = 0.0,
maxWidth = size.width,
minHeight = 0.0,
maxHeight = size.height;
Example 3 では Center
をにより、表示領域より小さい、という制約が渡されました。
つまり、Looseな制約がContainer
に渡されたということです。
結局の所、ここでのCenter
の役割は、親から渡されたTightな制約を、
Looseな制約に変換し子に渡すことです。
まとめ
本記事では、Flutter 公式のウィジェットの大きさ等に関するドキュメント、
Understanding Constraints を日本語で解説しました。
記事内で、様々な実装例をコードと横並びで確認できるWEBアプリを紹介しました。
いかがだったでしょうか?
ウィジェットのレイアウトの仕方が少しでもイメージできるようになったのであれば、
とても嬉しいです。
ぜひ本記事、また、元記事を繰り返し読んで、
理解を深めていただければと思います。
本記事があなたのアプリ開発の一助となれば幸いです。
編集後記(2022/12/27)
ウィジェットのレイアウトの仕方についての記事でした。
ウィジェットのレイアウトの仕方については、
以下の記事が根本のところからかなり詳しく解説されています。
併せてご確認いただければと思います。
今回紹介したWEBアプリのGitHubリポジトリは以下となります。
レスポンシブ対応等を結構頑張って実装しましたので、
ぜひ見ていただければ嬉しいです。
記事のためのアプリを作る形式は、
なかなか骨が折れるものでした。
好評だったら今後もやっていこうと思いますので、
ぜひFlutter大学のSlackやTwitter等で感想をいただければと思います。
以上、読んでいただきありがとうございました。
週刊Flutter大学では、Flutterに関する技術記事、Flutter大学についての紹介記事を投稿していきます。
記事の更新情報はFlutter大学Twitterにて告知します。
ぜひぜひフォローをお願いいたします。