আপনার রচনা UI আর্কিটেক্ট করা হচ্ছে

রচনায় UI অপরিবর্তনীয়—এটি আঁকার পরে এটি আপডেট করার কোনো উপায় নেই। আপনি যা নিয়ন্ত্রণ করতে পারেন তা হল আপনার UI এর অবস্থা। প্রতিবার UI এর অবস্থা পরিবর্তিত হলে, কম্পোজ UI ট্রির পরিবর্তিত অংশগুলিকে পুনরায় তৈরি করে । কম্পোজেবল স্টেট গ্রহণ করতে পারে এবং ইভেন্টগুলি প্রকাশ করতে পারে-উদাহরণস্বরূপ, একটি TextField একটি মান গ্রহণ করে এবং onValueChange একটি কলব্যাক প্রকাশ করে যা কলব্যাক হ্যান্ডলারকে মান পরিবর্তন করার জন্য অনুরোধ করে।

var name by remember { mutableStateOf("") }
OutlinedTextField(
    value = name,
    onValueChange = { name = it },
    label = { Text("Name") }
)

যেহেতু কম্পোজেবলগুলি স্টেট গ্রহণ করে এবং ঘটনাগুলি প্রকাশ করে, তাই একমুখী ডেটা প্রবাহ প্যাটার্ন জেটপ্যাক কম্পোজের সাথে ভালভাবে ফিট করে। এই নির্দেশিকাটি কম্পোজে কীভাবে একমুখী ডেটা প্রবাহের প্যাটার্ন বাস্তবায়ন করতে হয়, কীভাবে ইভেন্ট এবং স্টেট হোল্ডারগুলিকে বাস্তবায়ন করতে হয় এবং কম্পোজে ভিউমডেলগুলির সাথে কীভাবে কাজ করতে হয় তার উপর ফোকাস করে৷

একমুখী তথ্য প্রবাহ

একটি ইউনিডাইরেশনাল ডাটা ফ্লো (UDF) হল একটি ডিজাইন প্যাটার্ন যেখানে স্টেট নিচে প্রবাহিত হয় এবং ইভেন্টগুলি উপরে প্রবাহিত হয়। একমুখী ডেটা ফ্লো অনুসরণ করে, আপনি কম্পোজেবলগুলিকে ডিকপল করতে পারেন যা আপনার অ্যাপের অংশগুলি থেকে UI-তে স্থিতি দেখায় যা সংরক্ষণ করে এবং অবস্থা পরিবর্তন করে।

একমুখী ডেটা প্রবাহ ব্যবহার করে একটি অ্যাপের জন্য UI আপডেট লুপ দেখতে এইরকম:

  • ইভেন্ট : UI এর অংশ একটি ইভেন্ট তৈরি করে এবং এটিকে ঊর্ধ্বমুখী করে, যেমন একটি বোতাম ক্লিক ভিউমডেল পরিচালনা করার জন্য পাস করা হয়; অথবা আপনার অ্যাপের অন্যান্য স্তর থেকে একটি ইভেন্ট পাস করা হয়েছে, যেমন ব্যবহারকারীর সেশনের মেয়াদ শেষ হয়ে গেছে।
  • আপডেট অবস্থা : একটি ইভেন্ট হ্যান্ডলার রাষ্ট্র পরিবর্তন করতে পারে।
  • ডিসপ্লে স্টেট : স্টেট হোল্ডার স্টেট থেকে চলে যায় এবং UI এটি প্রদর্শন করে।

চিত্র 1. একমুখী তথ্য প্রবাহ।

জেটপ্যাক কম্পোজ ব্যবহার করার সময় এই প্যাটার্ন অনুসরণ করা বিভিন্ন সুবিধা প্রদান করে:

  • পরীক্ষাযোগ্যতা : UI থেকে ডিকপলিং স্টেট যা প্রদর্শন করে তা বিচ্ছিন্নভাবে উভয় পরীক্ষা করা সহজ করে তোলে।
  • স্টেট এনক্যাপসুলেশন : যেহেতু স্টেট শুধুমাত্র একটি জায়গায় আপডেট করা যায় এবং একটি কম্পোজেবল স্টেটের জন্য সত্যের একটি মাত্র উৎস আছে, তাই অসামঞ্জস্যপূর্ণ অবস্থার কারণে আপনি বাগ তৈরি করার সম্ভাবনা কম।
  • UI সামঞ্জস্যতা : StateFlow বা LiveData এর মতো পর্যবেক্ষণযোগ্য স্টেট হোল্ডার ব্যবহার করে সমস্ত স্টেট আপডেট অবিলম্বে UI-তে প্রতিফলিত হয়।

জেটপ্যাক রচনায় একমুখী ডেটা প্রবাহ

কম্পোজেবলগুলি রাষ্ট্র এবং ঘটনাগুলির উপর ভিত্তি করে কাজ করে। উদাহরণস্বরূপ, একটি TextField শুধুমাত্র তখনই আপডেট হয় যখন এর value প্যারামিটার আপডেট করা হয় এবং এটি একটি onValueChange কলব্যাক প্রকাশ করে—একটি ইভেন্ট যা মানটিকে একটি নতুন তে পরিবর্তন করার অনুরোধ করে। কম্পোজ State অবজেক্টকে একটি মান ধারক হিসাবে সংজ্ঞায়িত করে এবং রাষ্ট্রীয় মানের পরিবর্তন একটি পুনর্গঠনকে ট্রিগার করে। আপনি কতক্ষণের জন্য মানটি মনে রাখতে হবে তার উপর নির্ভর করে আপনি একটি remember { mutableStateOf(value) } বা একটি rememberSaveable { mutableStateOf(value) তে স্টেট ধরে রাখতে পারেন৷

TextField কম্পোজেবলের মান হল String , তাই এটি যেকোনো জায়গা থেকে আসতে পারে—হার্ডকোড করা মান থেকে, ভিউমডেল থেকে, অথবা প্যারেন্ট কম্পোজেবল থেকে পাস করা যেতে পারে। আপনাকে এটিকে State অবজেক্টে ধরে রাখতে হবে না, তবে যখন onValueChange কল করা হয় তখন আপনাকে মানটি আপডেট করতে হবে।

কম্পোজেবল প্যারামিটার সংজ্ঞায়িত করুন

একটি কম্পোজেবলের স্টেট প্যারামিটারগুলি সংজ্ঞায়িত করার সময় আপনাকে নিম্নলিখিত প্রশ্নগুলি মনে রাখা উচিত:

  • কম্পোজেবল কতটা পুনর্ব্যবহারযোগ্য বা নমনীয়?
  • কিভাবে রাষ্ট্র পরামিতি এই কম্পোজেবল এর কর্মক্ষমতা প্রভাবিত করে?

ডিকপলিং এবং পুনঃব্যবহারকে উত্সাহিত করার জন্য, প্রতিটি কম্পোজেবলের সম্ভাব্য সর্বনিম্ন পরিমাণ তথ্য রাখা উচিত। উদাহরণস্বরূপ, একটি সংবাদ নিবন্ধের শিরোনাম ধরে রাখার জন্য একটি কম্পোজেবল তৈরি করার সময়, সম্পূর্ণ সংবাদ নিবন্ধের পরিবর্তে শুধুমাত্র সেই তথ্যগুলিকে পাস করতে পছন্দ করুন যা প্রদর্শন করা প্রয়োজন:

@Composable
fun Header(title: String, subtitle: String) {
    // Recomposes when title or subtitle have changed.
}

@Composable
fun Header(news: News) {
    // Recomposes when a new instance of News is passed in.
}

কখনও কখনও, স্বতন্ত্র পরামিতি ব্যবহার করা কর্মক্ষমতাকেও উন্নত করে-উদাহরণস্বরূপ, যদি News শুধুমাত্র title এবং subtitle ছাড়া আরও তথ্য থাকে, যখনই News এর একটি নতুন উদাহরণ Header(news) -এ পাস করা হয়, তাহলে কম্পোজেবলটি পুনরায় কম্পোজ করবে, এমনকি title এবং subtitle না থাকলেও পরিবর্তিত

আপনার পাস করা প্যারামিটারের সংখ্যা সাবধানতার সাথে বিবেচনা করুন৷ অনেকগুলি পরামিতি সহ একটি ফাংশন থাকলে ফাংশনের ergonomics হ্রাস পায়, তাই এই ক্ষেত্রে সেগুলিকে একটি শ্রেণিতে গোষ্ঠীবদ্ধ করা পছন্দনীয়৷

রচনা ইভেন্ট

আপনার অ্যাপের প্রতিটি ইনপুট একটি ইভেন্ট হিসাবে উপস্থাপন করা উচিত: ট্যাপ, পাঠ্য পরিবর্তন এবং এমনকি টাইমার বা অন্যান্য আপডেট। যেহেতু এই ইভেন্টগুলি আপনার UI এর অবস্থাকে পরিবর্তন করে, তাই ViewModel হতে হবে সেগুলিকে পরিচালনা করবে এবং UI অবস্থা আপডেট করবে৷

UI স্তরটি কখনই কোনও ইভেন্ট হ্যান্ডলারের বাইরের অবস্থা পরিবর্তন করা উচিত নয় কারণ এটি আপনার অ্যাপ্লিকেশনে অসঙ্গতি এবং বাগগুলি প্রবর্তন করতে পারে।

রাষ্ট্র এবং ইভেন্ট হ্যান্ডলার ল্যাম্বডাসের জন্য অপরিবর্তনীয় মান পাস করা পছন্দ করুন। এই পদ্ধতির নিম্নলিখিত সুবিধা রয়েছে:

  • আপনি পুনর্ব্যবহারযোগ্যতা উন্নত করুন।
  • আপনি নিশ্চিত করুন যে আপনার UI সরাসরি রাষ্ট্রের মান পরিবর্তন করে না।
  • আপনি কনকারেন্সি সমস্যাগুলি এড়িয়ে যান কারণ আপনি নিশ্চিত করেন যে রাজ্যটি অন্য থ্রেড থেকে মিউট করা হয়নি।
  • প্রায়শই, আপনি কোড জটিলতা হ্রাস করেন।

উদাহরণস্বরূপ, একটি কম্পোজেবল যা একটি String এবং একটি ল্যাম্বডাকে পরামিতি হিসাবে গ্রহণ করে অনেক প্রসঙ্গ থেকে কল করা যেতে পারে এবং এটি অত্যন্ত পুনঃব্যবহারযোগ্য। ধরুন আপনার অ্যাপের উপরের অ্যাপ বারে সবসময় টেক্সট দেখায় এবং একটি ব্যাক বোতাম থাকে। আপনি আরও জেনেরিক MyAppTopAppBar কম্পোজেবল সংজ্ঞায়িত করতে পারেন যা পাঠ্য এবং ব্যাক বোতাম হ্যান্ডেলকে পরামিতি হিসাবে গ্রহণ করে:

@Composable
fun MyAppTopAppBar(topAppBarText: String, onBackPressed: () -> Unit) {
    TopAppBar(
        title = {
            Text(
                text = topAppBarText,
                textAlign = TextAlign.Center,
                modifier = Modifier
                    .fillMaxSize()
                    .wrapContentSize(Alignment.Center)
            )
        },
        navigationIcon = {
            IconButton(onClick = onBackPressed) {
                Icon(
                    Icons.AutoMirrored.Filled.ArrowBack,
                    contentDescription = localizedString
                )
            }
        },
        // ...
    )
}

মডেল, রাজ্য এবং ঘটনা দেখুন: একটি উদাহরণ

ViewModel এবং mutableStateOf ব্যবহার করে, আপনি আপনার অ্যাপে একমুখী ডেটা প্রবাহ চালু করতে পারেন যদি নিম্নলিখিতগুলির মধ্যে একটি সত্য হয়:

  • StateFlow বা LiveData এর মতো পর্যবেক্ষণযোগ্য স্টেট হোল্ডারের মাধ্যমে আপনার UI-এর অবস্থা প্রকাশ করা হয়।
  • ViewModel আপনার অ্যাপের UI বা অন্যান্য স্তর থেকে আসা ইভেন্টগুলি পরিচালনা করে এবং ইভেন্টগুলির উপর ভিত্তি করে স্টেট হোল্ডার আপডেট করে৷

উদাহরণস্বরূপ, একটি সাইন-ইন স্ক্রীন প্রয়োগ করার সময়, একটি সাইন ইন বোতামে আলতো চাপলে আপনার অ্যাপটি একটি অগ্রগতি স্পিনার এবং একটি নেটওয়ার্ক কল প্রদর্শন করবে৷ যদি লগইন সফল হয়, তাহলে আপনার অ্যাপটি একটি ভিন্ন স্ক্রিনে নেভিগেট করবে; কোনো ত্রুটির ক্ষেত্রে অ্যাপটি একটি স্ন্যাকবার দেখায়। আপনি কীভাবে পর্দার অবস্থা এবং ইভেন্টের মডেল করবেন তা এখানে:

পর্দার চারটি অবস্থা রয়েছে:

  • সাইন আউট : যখন ব্যবহারকারী এখনও সাইন ইন করেননি।
  • প্রগতিতে : যখন আপনার অ্যাপ বর্তমানে একটি নেটওয়ার্ক কল করে ব্যবহারকারীকে সাইন ইন করার চেষ্টা করছে।
  • ত্রুটি : যখন সাইন ইন করার সময় একটি ত্রুটি ঘটেছে৷
  • সাইন ইন করেছেন : যখন ব্যবহারকারী সাইন ইন করেন।

আপনি একটি সিল ক্লাস হিসাবে এই রাজ্যগুলি মডেল করতে পারেন. ViewModel রাজ্যটিকে একটি State হিসাবে প্রকাশ করে, প্রাথমিক অবস্থা সেট করে এবং প্রয়োজন অনুসারে রাষ্ট্রকে আপডেট করে। ViewModel এছাড়াও একটি onSignIn() পদ্ধতি প্রকাশ করে সাইন-ইন ইভেন্ট পরিচালনা করে।

class MyViewModel : ViewModel() {
    private val _uiState = mutableStateOf<UiState>(UiState.SignedOut)
    val uiState: State<UiState>
        get() = _uiState

    // ...
}

mutableStateOf API ছাড়াও, কম্পোজ LiveData , Flow , এবং Observable শ্রোতা হিসাবে নিবন্ধন করতে এবং একটি রাষ্ট্র হিসাবে মান উপস্থাপন করার জন্য এক্সটেনশন সরবরাহ করে

class MyViewModel : ViewModel() {
    private val _uiState = MutableLiveData<UiState>(UiState.SignedOut)
    val uiState: LiveData<UiState>
        get() = _uiState

    // ...
}

@Composable
fun MyComposable(viewModel: MyViewModel) {
    val uiState = viewModel.uiState.observeAsState()
    // ...
}

আরও জানুন

জেটপ্যাক কম্পোজে আর্কিটেকচার সম্পর্কে আরও জানতে, নিম্নলিখিত সংস্থানগুলি দেখুন:

নমুনা

{% শব্দার্থে %} {% endverbatim %} {% শব্দার্থে %} {% endverbatim %}