গন্তব্যের মধ্যে অ্যানিমেট করুন

আপনার অ্যাপে ব্যবহারকারীরা নেভিগেট করার সময় মসৃণ ভিজ্যুয়াল ট্রানজিশন তৈরি করার জন্য NavDisplay বিল্ট-ইন অ্যানিমেশন সুবিধা রয়েছে। আপনি এই অ্যানিমেশনগুলো NavDisplay এর জন্য বিশ্বব্যাপী অথবা মেটাডেটা ব্যবহার করে Scene লেভেলে কাস্টমাইজ করতে পারেন।

অন্তর্নির্মিত অ্যানিমেশন ক্ষমতাগুলো বুঝুন

ন্যাভিগেশনের সময় কন্টেন্ট কীভাবে অ্যানিমেট হবে তা নির্ধারণ করতে NavDisplay ContentTransform API ব্যবহার করে। যখন বর্তমান সিন-এর ক্লাস এবং তার key প্রপার্টি থেকে প্রাপ্ত কোনো 'key' পরিবর্তিত হয়, তখন NavDisplay স্বয়ংক্রিয়ভাবে সিনগুলোর মধ্যে ট্রানজিশন অ্যানিমেট করে। এই 'key' পরিবর্তিত হলে, NavDisplay ট্রানজিশনের ধরন—যেমন ফরওয়ার্ড, ব্যাক, বা প্রেডিক্টিভ ব্যাক—এর জন্য উপযুক্ত সিন-এর ContentTransform ব্যবহার করে। যদি সেই ContentTransform সংজ্ঞায়িত না থাকে, তবে NavDisplay তার সংশ্লিষ্ট ডিফল্ট ট্রানজিশন ব্যবহার করে।

ডিফল্ট ট্রানজিশনগুলি ওভাররাইড করুন

আপনি NavDisplay তে ট্রানজিশন প্যারামিটার প্রদান করে ডিফল্ট অ্যানিমেশন আচরণগুলো পরিবর্তন করতে পারেন।

  • transitionSpec : এই প্যারামিটারটি নির্ধারণ করে যে, ব্যাক স্ট্যাকে কন্টেন্ট যুক্ত করার সময় (অর্থাৎ, সামনে নেভিগেট করার সময়) কোন ContentTransform প্রয়োগ করা হবে।
  • popTransitionSpec : এই প্যারামিটারটি নির্ধারণ করে যে, ব্যাক স্ট্যাক থেকে কন্টেন্ট সরিয়ে ফেলার সময় (অর্থাৎ, পিছনে নেভিগেট করার সময়) কোন ContentTransform প্রয়োগ করা হবে।
  • predictivePopTransitionSpec : এই প্যারামিটারটি নির্ধারণ করে যে, একটি প্রেডিক্টিভ ব্যাক জেসচার ব্যবহার করে কন্টেন্ট পপ করার সময় কোন ContentTransform প্রয়োগ করা হবে।

Scene স্তরে ট্রানজিশন ওভাররাইড করুন

আপনি NavDisplay দ্বারা সংজ্ঞায়িত নিম্নলিখিত মেটাডেটা কীগুলি ব্যবহার করে স্বতন্ত্র দৃশ্যের জন্য কাস্টম অ্যানিমেশন নির্ধারণ করতে পারেন:

প্রদান করা হলে, NavDisplay তে সেট করা সংশ্লিষ্ট ডিফল্টগুলোর পরিবর্তে এই সিন-লেভেল ট্রানজিশনগুলো ব্যবহৃত হয়।

নিম্নলিখিত কোড স্নিপেটটি গ্লোবাল NavDisplay ট্রানজিশন এবং স্বতন্ত্র NavEntry লেভেলে ওভাররাইড উভয়ই প্রদর্শন করে:

@Serializable
data object ScreenA : NavKey

@Serializable
data object ScreenB : NavKey

@Serializable
data object ScreenC : NavKey

class AnimatedNavDisplayActivity : ComponentActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContent {

            Scaffold { paddingValues ->

                val backStack = rememberNavBackStack(ScreenA)

                NavDisplay(
                    backStack = backStack,
                    onBack = { backStack.removeLastOrNull() },
                    entryProvider = entryProvider {
                        entry<ScreenA> {
                            ContentOrange("This is Screen A") {
                                Button(onClick = { backStack.add(ScreenB) }) {
                                    Text("Go to Screen B")
                                }
                            }
                        }
                        entry<ScreenB> {
                            ContentMauve("This is Screen B") {
                                Button(onClick = { backStack.add(ScreenC) }) {
                                    Text("Go to Screen C")
                                }
                            }
                        }
                        entry<ScreenC>(
                            metadata = metadata {
                                put(NavDisplay.TransitionKey) {
                                    // Slide new content up, keeping the old content in place underneath
                                    slideInVertically(
                                        initialOffsetY = { it },
                                        animationSpec = tween(1000)
                                    ) togetherWith ExitTransition.KeepUntilTransitionsFinished
                                }
                                put(NavDisplay.PopTransitionKey) {
                                    // Slide old content down, revealing the new content in place underneath
                                    EnterTransition.None togetherWith
                                            slideOutVertically(
                                                targetOffsetY = { it },
                                                animationSpec = tween(1000)
                                            )
                                }
                                put(NavDisplay.PredictivePopTransitionKey) {
                                    // Slide old content down, revealing the new content in place underneath
                                    EnterTransition.None togetherWith
                                            slideOutVertically(
                                                targetOffsetY = { it },
                                                animationSpec = tween(1000)
                                            )
                                }
                            }
                        ) {
                            ContentGreen("This is Screen C")
                        }
                    },
                    transitionSpec = {
                        // Slide in from right when navigating forward
                        slideInHorizontally(initialOffsetX = { it }) togetherWith
                            slideOutHorizontally(targetOffsetX = { -it })
                    },
                    popTransitionSpec = {
                        // Slide in from left when navigating back
                        slideInHorizontally(initialOffsetX = { -it }) togetherWith
                            slideOutHorizontally(targetOffsetX = { it })
                    },
                    predictivePopTransitionSpec = {
                        // Slide in from left when navigating back
                        slideInHorizontally(initialOffsetX = { -it }) togetherWith
                            slideOutHorizontally(targetOffsetX = { it })
                    },
                    modifier = Modifier.padding(paddingValues)
                )
            }
        }
    }
}

চিত্র ১। কাস্টম অ্যানিমেশন সহ অ্যাপ।

দৃশ্যগুলির মধ্যে ট্রানজিশন ন্যাভ এন্ট্রি

যেসব অ্যাপ সিন (scene) ব্যবহার করে কাস্টম লেআউট তৈরি করে , সেখানে ট্রানজিশনের সময় একটি NavEntry উভয় সিনেরই entries প্রপার্টিতে অন্তর্ভুক্ত থাকতে পারে। অভ্যন্তরীণভাবে, NavDisplay যাচাই করে যে প্রতিটি এন্ট্রি যেকোনো সময়ে সর্বাধিক একটি সিনে প্রদর্শিত হচ্ছে কিনা, যার ফলে যখন কোনো NavEntry রেন্ডারকারী সিন পরিবর্তিত হয়, তখন ট্রানজিশনটি ঝাঁকুনিপূর্ণ হতে পারে। সিনগুলোর মধ্যে এন্ট্রিগুলোকে মসৃণভাবে অ্যানিমেট করার জন্য, আপনি আপনার NavDisplay একটি SharedTransitionLayout মধ্যে র‍্যাপ করতে পারেন এবং নিম্নলিখিত উদাহরণে দেখানো অনুযায়ী NavDisplay তে SharedTransitionScope প্রদান করতে পারেন:

SharedTransitionLayout {
    NavDisplay(
        // ...
        sharedTransitionScope = this
    )
}