Upgrade to Pro — share decks privately, control downloads, hide ads and more …

Re:VIEWで書いた「Compose で Android の edge-to-edge に対...

Re:VIEWで書いた「Compose で Android の edge-to-edge に対応する」をRoo Codeで発表資料にしてもらった

Tomoya Miwa

April 02, 2025
Tweet

More Decks by Tomoya Miwa

Other Decks in Technology

Transcript

  1. Re:VIEW で書いた「Compose で Android の edge-to- edge に対応する」を Roo Code

    で発表資料にしても らった 2025/04/04 tomoya0x00 shibuya.apk#52 1
  2. Edge to edge opt-out going away Android 15 enforced edge-to-edge

    for apps targeting Android 15 (API level 35), but your app could opt-out by setting R.attr#windowOptOutEdgeToEdgeEnforcement to true. For apps targeting Android 16 (API level 36), R.attr#windowOptOutEdgeToEdgeEnforcement is deprecated and disabled, and your app can't opt-out of going edge-to-edge. https://developer.android.com/about/versions/16/behavior-changes-16 4
  3. ちょうど、技術同人誌「U-NEXT TECH STREAM 02」 で「第 3 章 : Compose で

    Android の edge-to-edge に 対応する」を執筆済みだった 5
  4. Roo Code って何? https://github.com/RooVetGit/Roo-Code Roo Code is an AI-powered autonomous

    coding agent that lives in your editor VS Code の拡張機能で、以前は Roo Cline という名前だった 以前の名前から分かるように、Cline の fork 版 OpenRouter などで API Key を生成して使うか、VS Code LM API 経由で GitHub Copilot 拡張機能が提供するモデルも使える(こちらは experimental) 今回使用したモデルは copilot - claude-3.5-sonnet コード生成だけではなく、コマンド実行してその結果を元に更にコード編集など もしてくれる 12
  5. 最初のプロンプト "Re:VIEW 記法で記述した android_edge_to_edge.re ファイルを元に 15 分間 LT の資料 を作成して下さい。資料の形式は

    Marp でお願いします。" ↓ これだけで、本当に Marp が解釈できる Markdown に変換してくれた!(少し間違って いるが画像も参照してくれている) ただし、15 分の LT にしては内容は少ない。 14
  6. 自己紹介 U-NEXT Android Engineer X: @tomoya0x00 2024 年は Android mobile/tablet

    アプリをフル Compose な新アプリに置き換え edge-to-edge 対応で得た知見を共有します 18
  7. なぜ今対応すべきか? Android 15 で edge-to-edge が強制される targetSdkVersion = 35 以上で必須に

    スマートフォンの前面がほぼ画面に より没入感の高いアプリ体験の提供が可能に Android 16 では一時的な回避も不可能に 20
  8. Android OS の進化 Android 10: ジェスチャーナビゲーション導入 Android 12: Material You、Dynamic

    Color Android 13: Predictive Back Gesture Android 14: より洗練されたシステム UI Android 15: edge-to-edge 強制化 Android 16: 回避オプション完全廃止 21
  9. 一時的な回避方法 Activity に適用する style に追記: <item name="android:windowOptOutEdgeToEdgeEnforcement">true</item> 注意点: Android 15(targetSdkVersion

    = 35)でのみ有効 Android 16(targetSdkVersion = 36)では使用不可 edge-to-edge 対応を先送りにするべきではない 早期の正式対応を強く推奨 22
  10. 基本的な対応の流れ ② 2. AndroidManifest.xml で Activity に設定追加 android:windowSoftInputMode="adjustResize" 3. 各

    Composable に対して適切な WindowInsets を指定 Modifier.windowInsetsPadding(WindowInsets.safeDrawing) 26
  11. WindowInsets の種類 よく使用する 3 種類: WindowInsets.safeDrawing システム UI でアプリの UI

    が隠れるのを防ぐ ボタンなどの操作可能な要素に使用 WindowInsets.safeGestures システムジェスチャとの衝突を防ぐ WindowInsets.safeContent 上記 2 つの組み合わせ 29
  12. WindowInsets の自動消費 親コンポーネントで消費された WindowInsets は子には適用されない: Box( modifier = Modifier .background(Color.Black)

    .fillMaxSize() .windowInsetsPadding(WindowInsets.safeDrawing) ) { Spacer( modifier = Modifier .background(Color.Gray) .fillMaxSize() .windowInsetsPadding(WindowInsets.safeDrawing) ) } 33
  13. WindowInsets の部分消費 親で一部の WindowInsets を消費した例: Box( modifier = Modifier .windowInsetsPadding(WindowInsets.statusBars)

    ) { Box( modifier = Modifier .windowInsetsPadding(WindowInsets.systemBars) ) { // statusBarsは適用されない // navigationBarsのみ適用される } } 35
  14. Material 2 での対応 ① TopAppBar の実装例: TopAppBar( windowInsets = WindowInsets.safeDrawing.only(

    WindowInsetsSides.Top + WindowInsetsSides.Horizontal ), title = { Text("TopAppBar") } ) 39
  15. Material 2 での対応 ② BottomNavigation の実装例: BottomNavigation( modifier = Modifier.fillMaxWidth(),

    windowInsets = WindowInsets.safeDrawing.only( WindowInsetsSides.Horizontal + WindowInsetsSides.Bottom ) ) { // ナビゲーションアイテム } 40
  16. Material 2 での対応 ③ NavigationRail の実装例: NavigationRail( modifier = Modifier.fillMaxHeight(),

    windowInsets = WindowInsets.safeDrawing.only( WindowInsetsSides.Vertical + WindowInsetsSides.Start ) ) { // ナビゲーションアイテム } 41
  17. ハマりどころ ②:LazyColumn リストの最後のアイテムが操作不能になる問題: LazyColumn { items(items) { item -> ListItem(item)

    } item { Spacer( modifier = Modifier .fillMaxWidth() .windowInsetsPadding( WindowInsets.safeDrawing.only( WindowInsetsSides.Bottom ) ) .focusable() ) } } 43
  18. TextField 付き LazyColumn 推奨実装: LazyColumn { items(items) { item ->

    TextField( value = item.text, onValueChange = { /* ... */ }, modifier = Modifier.fillMaxWidth() ) } item { Spacer( modifier = Modifier .windowInsetsPadding( WindowInsets.ime.add( WindowInsets.safeDrawing.only( WindowInsetsSides.Bottom ) ) ) .focusable() ) 44
  19. 二重適用の解決策 ① consumeWindowInsets の使用: Column(modifier = Modifier.fillMaxSize()) { TopAppBar( windowInsets

    = WindowInsets.safeDrawing.only( WindowInsetsSides.Top + WindowInsetsSides.Horizontal ), title = { Text("TopAppBar") } ) Box( modifier = Modifier .consumeWindowInsets( WindowInsets.safeDrawing.only( WindowInsetsSides.Top ) ) .windowInsetsPadding(WindowInsets.safeDrawing) ) { // コンテンツ 46
  20. 二重適用の解決策 ② 必要な WindowInsets のみを指定: Column(modifier = Modifier.fillMaxSize()) { TopAppBar(

    windowInsets = WindowInsets.safeDrawing.only( WindowInsetsSides.Top + WindowInsetsSides.Horizontal ), title = { Text("TopAppBar") } ) Box( modifier = Modifier .windowInsetsPadding( WindowInsets.safeDrawing.only( WindowInsetsSides.Bottom + WindowInsetsSides.Horizontal ) ) ) { // コンテンツ 47
  21. ハマりどころ ④:WindowInsets IgnoringVisibility 問題点: API Level 30 未満で正しく動作しない 古い Android

    ではナビゲーションバーの高さが取得できない // 安全な使用方法 Modifier.windowInsetsPadding( if (VERSION.SDK_INT >= VERSION_CODES.R) { WindowInsets.systemBarsIgnoringVisibility } else { WindowInsets.systemBars } ) 48
  22. IgnoringVisibility の問題点 API Level 29 以下では: 正確な値が取得できない 特に IME で問題が顕著

    API Level 23 未満では全く動作しない ナビゲーションバーの高さが 0 になる可能性 実装時は必ず API Level 依存を考慮する 49
  23. ハマりどころ ⑤:SplashScreen 問題: AndroidX SplashScreen 1.0.1 で setOnExitAnimationListener()を呼ぶと edge-to- edge

    が壊れる ステータスバーの透過が取り消される 対策: SplashScreen#setOnExitAnimationListener()の使用を避ける 1.2.0-alpha01 以降で修正予定 50
  24. ハマりどころ ⑥:IME と WindowInsets 問題: IME に関する WindowInsets が正しく適用されない 原因と解決策:

    AndroidManifest.xml での設定漏れ android:windowSoftInputMode="adjustResize" 特に古い Android(Android 7 以下)で問題が顕著 51
  25. ハマりどころ ⑦:消費済み WindowInsets 問題: 親 Composable で消費済みの WindowInsets が子 Composable

    に再度適用される 解決策: 1. asPaddingValues()の使用を避ける 2. Modifer.windowInsetsPadding を使用 3. MutableWindowInsets と Modifier.onConsumedWindowInsetsChanged を組み合わ せる 53
  26. Surface での実装例 注:Scaffold 内部ではほぼ同じ事をやっています by tomoya0x00 @Composable fun MySurface( content:

    @Composable (PaddingValues) -> Unit ) { val consumedWindowInsets = remember { MutableWindowInsets() } Surface( modifier = Modifier.onConsumedWindowInsetsChanged { consumedWindowInsets.insets = it }, content = { paddingValues -> Box( modifier = Modifier.windowInsetsPadding( WindowInsets.safeDrawing .exclude(consumedWindowInsets.insets) ) ) { content(paddingValues) } 54
  27. スクロール TopAppBar の実装 val scrollBehavior = remember { TopAppBarDefaults.exitUntilCollapsedScrollBehavior() }

    Scaffold( modifier = Modifier.nestedScroll(scrollBehavior.nestedScrollConnection), topBar = { TopAppBar( title = { Text("Scroll to Hide") }, scrollBehavior = scrollBehavior ) } ) { padding -> LazyColumn( modifier = Modifier.padding(padding) ) { // リストアイテム } } 56
  28. Navigation rail 実装例 @Composable fun AdaptiveNavigation( windowSizeClass: WindowSizeClass, content: @Composable

    () -> Unit ) { if (windowSizeClass.widthSizeClass >= WindowWidthSizeClass.Medium) { Row { NavigationRail( windowInsets = WindowInsets.safeDrawing.only( WindowInsetsSides.Vertical + WindowInsetsSides.Start ) ) { // ナビゲーションアイテム } content() } } else { // Bottom Navigation使用 } } 58
  29. テーブルトップモードの実装 ① ディスプレイカットアウト対応: Surface( modifier = Modifier.fillMaxSize(), windowInsets = if

    (isTableTopMode) { WindowInsets(0.dp) // カットアウトを無視 } else { WindowInsets.safeDrawing.only( WindowInsetsSides.Horizontal ) } ) { // コンテンツ } 60
  30. テーブルトップモードの実装 ② 画面分割の考慮: Column(modifier = Modifier.fillMaxSize()) { Box( modifier =

    Modifier .weight(1f) .fillMaxWidth() ) { // ビデオプレイヤー } Box( modifier = Modifier .height(240.dp) .fillMaxWidth() .windowInsetsPadding( WindowInsets.safeDrawing.only( WindowInsetsSides.Bottom ) ) ) { // コントローラー 61
  31. まとめ Android 15 で edge-to-edge 対応が必須に Android 16 では回避オプションも無効に WindowInsets

    の適切な管理が重要 Material 3 利用で楽に対応可能 ハマりどころに注意 より進んだ対応で UX を向上 書籍で更に詳しい解説をご覧いただけます! 64
  32. Roo Code で Re:VIEW から発表資料をつくってみて (1/2) Marp 自体については説明してないのに、一発で Markdown 変換してくれてすごい

    曖昧な指示(例:15 分 LT の発表資料にして)だと、アウトプットは曖昧になる もっと相談しながらやったり、具体的な指示を出した方が良い コードのスライドが一部見切れていたり、画像のチョイスが微妙だったりした 具体的に指示したり、手作業で調整した Markdown が複雑になってくると、修正に失敗する事もある 66
  33. Roo Code で Re:VIEW から発表資料をつくってみて (2/2) 元の Re:VIEW ファイルには存在しない記述もいくつかスライドに追加されている ざっと確認した限りは正しそう

    本の執筆時にも相談しながらやってみると、自分の知らない知識を得られる かも ハルシネーションには注意が必要 便利なので、有効活用していきたい 67