Android অ্যাপে একটা RecyclerView-তে একাধিক টাইপের View দেখানো
একই লিস্টের মধ্যে অনেক সময়েই বিভিন্ন ধরনের ডেটা শো করানোর দরকার হয়। সেইসব ভিন্ন ভিন্ন ধরনের ডেটার ক্লিক ইভেন্টও ভিন্ন হয়। যেমন ধরেন, ফেসবুকের টাইমলাইনে আমরা টেক্সট টাইপের পোস্ট, ইমেজ টাইপের পোস্ট, ভিডিও টাইপের পোস্ট আবার কখনো বিজ্ঞাপনের পোস্ট দেখতে পাই। একেক ধরনের পোস্টে ক্লিক করলে কিন্তু একেক ধরনের অ্যাকশন হয়। আজকের এই টিউটোরিয়ালে দেখব কিভাবে একটা RecyclerView-তে একাধিক ধরনের ভিউ শো করানো যায়।
লেখাটি প্রথম প্রকাশিত হয়েছে আমার ব্যক্তিগত ব্লগে। ব্লগ থেকে পড়তে ক্লিক করুন
এই পোস্টটি বুঝার জন্য আপনাকে আগে থেকে RecyclerView কিভাবে কাজ করে বা ReclyeclerView দিয়ে কিভাবে একটা লিস্ট শো করতে হয় সে ব্যাপারে জানা থাকতে হবে। আপনি যদি আগে RecyclerView এর সাথে পরিচিত না হয়ে থাকেন তাহলে আমার এই ব্লগ পোস্টটি দেখতে পারেন। এখানে CardView ও RecyclerView ব্যবহার করে Horizontal ও Vertical scrollable লিস্ট কিভাবে জেনারেট করতে হয় সেটা বিস্তারিত ব্যাখ্যা সহ দেখানো হয়েছে। তাই আজকের এই পোস্টে একই বিষয়গুলো রিপিট করা হবে না। আর টিউটোরিয়ালটার স্যাম্পল কোডগুলো দেখানো হবে Kotlin programming language ব্যবহার করে।
Problem Description
উপরের ছবির মত একটা timeline feed বানাতে হবে। অনেকটা ফেসবুকের মত হবে টাইমলাইনটা। আপাতত এই টাইমলাইনে শুধু ২ ধরনের ভিউ দেখা যাবে। প্রথমত, শুধু টেক্সট টাইপের পোস্ট দেখা যাবে। দ্বিতীয়ত, ইমেজ সহ পোস্ট দেখা যাবে। আপনি চাইলে প্র্যাকটিসের জন্য ভিডিও টাইপ পোস্ট ও বিজ্ঞাপন দেখানোর জন্য আলাদা ভিউ রাখতে পারেন। টিউটোরিয়ালটা সিম্পল রাখার জন্য উভয় টাইপের ভিউতে ক্লিক করলে আলাদা আলাদা Toast message দেখাবে। উল্লেখ্য, এই টাইমলাইনটা বানাতে হবে একটা মাত্র RecyclerView ব্যবহার করে।
Solution
আমরা সাধারণত RecyclerView এর জন্য একটা custom Adapter ও একটা custom ViewHolder class বানাই। Adapter class এর ভিতর আমরা ViewHolder এ ডেটা populate করি ও ক্লিক ইভেন্ট নিয়ে কাজ করি। এই way তে কাজ করা হয়ে থাকে, সাধারণত যখন একটা লিস্টে এক ধরনের ডেটাই শো করতে চাই এবং সকল আইটেমের ক্লিক ইভেন্টও একই রকমের হয়। কিন্তু আমাদের প্রবলেমটা ভিন্ন। আমাদের লিস্টের সকল ভিউ এক রকম নয়। তাদের ক্লিক ইভেন্টও এক রকম নয়। আমাদের দরকার ২ রকমের ভিউ দেখানো আর ২ ভাবে তাদের ক্লিক ইভেন্ট হ্যান্ডেল করা।
Data Class
Timeline এর প্রতিটা পোস্টের ডেটা hold করার জন্য একটা Kotlin Data Class বানাই।
data class PostData(
val userName: String,
val userProfilePhotoUrl: String,
val timeStamp: String,
val postDescription: String,
val postImageUrl: String
)
RecylerView-তে PostData ক্লাসের একটা লিস্ট শো করা হবে। একই ডেটা মডেল দিয়ে আমরা টেক্সট টাইপ পোস্ট আর ইমেজ টাইপ পোস্ট দুইটাই হ্যান্ডেল করব। এজন্য লজিক ব্যবহার করছি এরকম যে, যদি postImageUrl স্ট্রিং এর ভ্যালু empty হয় তাহলে ধরে নিব সেটা টেক্সট টাইপের পোস্ট। আর যদি postImageUrl এর ভ্যালু empty না হয় তাহলে ধরে নিব এটা একটা Image type post. আপনার প্রোজেক্টর চাহিদা অনুযায়ী অন্য লজিক দিয়ে ভিউগুলোকে uniquely identify করতে পারেন। এমন কি আপনার ডেটা ক্লাসে সরাসরি postType নামের ভ্যারিয়েবলও রাখতে পারেন যা কিনা সার্ভার থেকে বা অন্য ডেটা সোর্স থেকে আসবে।
ViewHolder Classes
আমরা যেহেতু ২ রকমের ভিউ নিয়ে কাজ করছি তাই দুইটা আলাদা custom ViewHolder ক্লাস বানিয়ে ফেলি।
class TextPostViewHolder(itemView: View) : RecyclerView.ViewHolder(itemView) { val profilePhoto = itemView.profilePhoto
val profileName = itemView.profileName
val timeStamp = itemView.timeStamp
val postDescription = itemView.postDescription}class ImagePostViewHolder(itemView: View) : RecyclerView.ViewHolder(itemView) { val profilePhoto = itemView.profilePhoto
val profileName = itemView.profileName
val timeStamp = itemView.timeStamp
val postDescription = itemView.postDescription
val imageView = itemView.imageView
}
দুইটা ViewHolder এর জন্য দুইটি xml ফাইলে আলাদা আলাদা করে UI design করে রেখেছি। ইমেজ টাইপের পোস্টে টেক্সট টাইপের পোস্টের সবগুলো component ই আছে। ইমেজের ভিউটা বাড়তি আছে।
RecyclerView Adapter Class
চলুন এবার দেখে নিই Custom Adapter ক্লাসটা।
class TimelineRecyclerViewAdapter(private val postDataList : List<PostData>): RecyclerView.Adapter<RecyclerView.ViewHolder>() { private val POST_TYPE_TEXT = 1
private val POST_TYPE_IMAGE = 2 override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): RecyclerView.ViewHolder {
val view: View
return if (viewType == POST_TYPE_TEXT) {
view = LayoutInflater.from(parent.context).inflate(R.layout.item_text_post, parent, false) TextPostViewHolder(view) //object of TextPostViewHolder will return
} else {
view = LayoutInflater.from(parent.context).inflate(R.layout.item_image_post, parent, false) ImagePostViewHolder(view) //object of ImagePostViewHolder will return
}
} override fun onBindViewHolder(holder: RecyclerView.ViewHolder, position: Int) { val postData = postDataList[position]
val context = holder.itemView.context if (holder.itemViewType == POST_TYPE_TEXT) {
val viewHolder = holder as TextPostViewHolder viewHolder.profileName.text = postData.userName
Glide.with(context).load(postData.userProfilePhotoUrl).into(viewHolder.profilePhoto)
viewHolder.timeStamp.text = postData.timeStamp
viewHolder.postDescription.text = postData.postDescription viewHolder.itemView.setOnClickListener {
Toast.makeText(context, "Text type post", Toast.LENGTH_SHORT).show()
} } else {
val viewHolder = holder as ImagePostViewHolder viewHolder.profileName.text = postData.userName
Glide.with(context).load(postData.userProfilePhotoUrl).into(viewHolder.profilePhoto)
viewHolder.timeStamp.text = postData.timeStamp
viewHolder.postDescription.text = postData.postDescription Glide.with(holder.itemView.context).load(postData.postImageUrl).into(viewHolder.imageView) viewHolder.itemView.setOnClickListener {
Toast.makeText(context, "Image type post", Toast.LENGTH_SHORT).show()
}
}
} override fun getItemCount(): Int {
return postDataList.size
} override fun getItemViewType(position: Int): Int {
return if (postDataList[position].postImageUrl.isEmpty()) POST_TYPE_TEXT else POST_TYPE_IMAGE
}
}
TimelineRecyclerViewAdapter ক্লাসের constructor এ PostData ক্লাসের একটা লিস্ট পাঠানো হয়েছে। এরপর ক্লাসের শুরুতে আমরা দুইটা int variable declare করেছি দুই রকমের ভিউ বুঝানোর জন্য।
আমরা জানি onCreateViewHolder() মেথড আমাদের RecyclerView এর view generate করে রিটার্ন করে। যখন লিস্টে একই রকম ভিউ দেখাতে চাই তখন এই মেথডের ভিতর আমাদের ভিউয়ের জন্য বানানো xml layout ফাইলটা inflate করে View class এর একটা অবজেক্ট বানাই। এরপর সেই ভিউ দিয়ে custom ViewHolder ক্লাসের একটা অবজেক্ট রিটার্ন করে দিই। কিন্তু আমাদের এই প্রবলেমের ক্ষেত্রে ২ রকমের ভিউ। তাই onCreateViewHolder() মেথডে আমরা view type চেক করেছি। সেই অনুযায়ী সংশ্লিষ্ট view holder class এর অবজেক্ট রিটার্ন করবে।
একই ভাবে onBindViewHolder() মেথডে ভিউ টাইপ চেক করে সে অনুযায়ী ভিউতে ডেটা populate করা হয়েছে। একই সাথে এখান থেকেই click event handle করা হয়েছে।
সবশেষে নতুন একটা মেথড দেখা যাচ্ছে getItemViewType(). এর কাজ কী? এর কাজ হচ্ছে প্রতিটা ভিউয়ের টাইপ কী সেটা রিটার্ন করা। এই মেথডের থেকে রিটার্ন করা ভ্যালুই মূলত আমরা উপরের দুইটা মেথডে পেয়েছি। getItemViewType() মেথডে আমরা চেক করছি পোস্টের ইমেজ url empty কিনা। যদি empty হয় তাহলে POST_TYPE_TEXT রিটার্ন করা হয়েছে অন্যথায় রিটার্ন হচ্ছে POST_TYPE_IMAGE. এই ভ্যারিয়েবল দুটি adapter class এর শুরুতেই declare করে রাখা হয়েছে। কোন লজিক ব্যবহার করে আপনি আপনার ভিউগুলোকে uniquely identify করবেন সেই লজিক এই override করা মেথডের ভিতর লিখতে হবে।
আশা করি বিষয়টা বুঝে ফেলেছেন। ভিন্ন রকমের ভিউ আরেকটা পারপাসে ইউজ করা যেতে পারে। যদি লিস্টে header বা footer থাকে তাহলে সেটাকে আলাদা ভিউ হিসাবে লিস্টে দেখাতে পারি। পুরো প্রোজেক্টের সোর্সকোড একত্রে পাওয়া যাবে আমার গিটহাব রিপোজিটরিতে। ব্লগ সম্পর্কে আপনার যে কোনো পরামর্শ বা গঠনমূলক সমালোচনা একান্ত কাম্য।