简体   繁体   English

将 RecyclerView 转换为 ViewPager/PagerAdapter

[英]Converting RecyclerView to ViewPager / PagerAdapter

Background - New to Android, but pretty nifty with moving layouts around, understanding Java, Kotlin and XML. However this task seems to be way above my head.背景- Android 的新手,但移动布局非常漂亮,了解 Java、Kotlin 和 XML。然而,这项任务似乎超出了我的理解范围。

Problem - I'm looking to convert the following Java file (RecyclerView) into a Kotlin file (ViewPager) - since I already have a ViewPager hooked-up with the same scrolling behaviour as desired.问题- 我希望将以下 Java 文件 (RecyclerView) 转换为 Kotlin 文件 (ViewPager) - 因为我已经连接了一个 ViewPager,其滚动行为与所需的相同。 I get the impression it's a 10min job for a seasoned developer.我的印象是,对于经验丰富的开发人员来说,这是一项 10 分钟的工作。 If that's the case I wonder if I could call upon some assistance from the community?如果是这样的话,我想知道我是否可以寻求社区的帮助? At least to work out where to start.至少要弄清楚从哪里开始。 I can't seem to find a guide on how to convert a RecyclerView into a PagerAdapter or ViewPager.我似乎找不到有关如何将 RecyclerView 转换为 PagerAdapter 或 ViewPager 的指南。

Essentially the existing ViewPager I'm using has static data (5 items) and this one could have tens of items, so needs to be dynamic with a separate datasource (items of CardItem).本质上,我正在使用的现有 ViewPager 有 static 个数据(5 个项目),而这个可能有数十个项目,因此需要动态使用单独的数据源(CardItem 的项目)。


RecyclerViewAdapter - Current (Java) RecyclerViewAdapter - 当前 (Java)

package com.APPNAME.fragments.cards;
import android.databinding.DataBindingUtil;
import android.support.v7.widget.RecyclerView;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import java.util.ArrayList;
import com.APPNAME.R;
import com.APPNAME.databinding.FragmentCardsRecentlyViewedBinding;
import com.APPNAME.model.cardItem.CardItem;



public class RecentlyViewedAdapter extends RecyclerView.Adapter<RecentlyViewedAdapter.RecentlyViewedViewHolder> {

    public OnCardClicked listener;
    private ArrayList<CardItem> items = new ArrayList<>();

    public void addItems(ArrayList<CardItem> list) {
        items = list;
        notifyDataSetChanged();
    }

    @Override
    public RecentlyViewedViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        FragmentCardsRecentlyViewedBinding binding = DataBindingUtil.inflate(LayoutInflater.from(parent.getContext()), R.layout.fragment_cards_recently_viewed, parent, false);
        return new RecentlyViewedViewHolder(binding);
    }

    @Override
    public void onBindViewHolder(RecentlyViewedViewHolder holder, int position) {
        holder.binding.setViewModel(new RecentlyViewedViewModel(items.get(position)));
    }

    @Override
    public int getItemCount() {
        return items.size();
    }

    interface OnCardClicked {
        void onCardClicked(View view, CardItem cardItem);
    }

    class RecentlyViewedViewHolder extends RecyclerView.ViewHolder {
        FragmentCardsRecentlyViewedBinding binding;

        public RecentlyViewedViewHolder(FragmentCardsRecentlyViewedBinding itemView) {
            super(itemView.getRoot());
            binding = itemView;
            binding.cardView.setOnClickListener(v -> {
                if (listener != null) {
                    listener.onCardClicked(v, items.get(getAdapterPosition()));
                }
            });
        }
    }
}

ViewPagerAdapter - Future (Kotlin) ViewPagerAdapter - 未来 (Kotlin)

package com.APPNAME.fragments.cards
import android.support.annotation.DrawableRes
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import java.util.*
import kotlinx.android.synthetic.main.fragment_cards_recently_viewed.view.*
import com.APPNAME.R
import com.APPNAME.activities.BaseActivity
import com.APPNAME.activities.cards.NewCardActivity
import com.APPNAME.model.cardItem.CardItem
import com.APPNAME.views.wrapContentViewPager.ObjectAtPositionPagerAdapter


class RecentlyViewedItemAdapter constructor(private val activity: BaseActivity) : ObjectAtPositionPagerAdapter() {

    private var items = ArrayList<CardItem>()

    override fun instantiateItemObject(container: ViewGroup, position: Int) : Any {
    return getImageView(container, R.drawable.placeholder_card_image) { NewCardActivity.start(activity, it) }
    }

    private fun getImageView(container: ViewGroup, @DrawableRes imageResourceId: Int, onClick: (imageResourceId: Int) -> Unit = {}): View {
    val layoutInflater = LayoutInflater.from(container.context)
        val layout = layoutInflater.inflate(R.layout.fragment_cards_recently_viewed, container, false)
        val image = layout.recentlyViewedImage
        image.setImageResource(imageResourceId)
        image.setOnClickListener { onClick(imageResourceId) }
        container.addView(layout)
        return layout
    }

    override fun isViewFromObject(view: View, anObject: Any) = (view == anObject)
    override fun getCount() = 5     //Placeholder

    override fun destroyItemObject(container: ViewGroup, position: Int, view: Any) {
        container.removeView(view as View)
    }
}

If you have relatively large number of items and need views to be recycled then the RecyclerView is the right option for you. 如果您有相对大量的物品并且需要回收视图,那么RecyclerView是您的正确选择。 It will be definitely easier than creating a custom view pager adapter with a fixed item count and a view holder pattern. 它肯定比使用固定项目计数和视图持有者模式创建自定义视图寻呼机适配器更容易。 The default/generic PageAdapter is not recycling anything - it's added to make sure you have all your views initiated and ready to swipe through. 默认/通用PageAdapter不会回收任何内容 - 它会被添加以确保您已启动所有视图并准备轻扫。 However, you can use https://developer.android.com/reference/android/support/v4/app/FragmentStatePagerAdapter.html that will destroy/recreate the fragments when no longer used or reused. 但是,您可以使用https://developer.android.com/reference/android/support/v4/app/FragmentStatePagerAdapter.html ,它将在不再使用或重用时销毁/重新创建片段。

Job done. 任务完成。 I actually surprised myself by being able to modify the recycler view to mimmick the PageView's behaviour precisely (including padding, snap scrolling etc). 我真的很惊讶自己能够修改回收器视图以精确地模仿PageView的行为(包括填充,快照滚动等)。 Thanks to the simple subclass SnapHelper . 感谢简单的子类SnapHelper Saved me a tonne of work, so no need to refactor the codebase. 为我节省了大量工作,因此无需重构代码库。

SnapHelper snapHelper = new PagerSnapHelper();
snapHelper.attachToRecyclerView(recyclerView);

PagerSnapHelper solved my problem. PagerSnapHelper 解决了我的问题。

   binding.recyclerViewHomeAnnouncement.apply {
        layoutManager = LinearLayoutManager(activity, LinearLayoutManager.HORIZONTAL, false)
        binding.recyclerViewHomeAnnouncement.layoutManager = layoutManager
        setHasFixedSize(true)
        itemAnimator = DefaultItemAnimator()
        adapter = homeAnnouncementAdapter
    }
    
    val snapHelper: SnapHelper = PagerSnapHelper()
    snapHelper.attachToRecyclerView(binding.recyclerViewHomeAnnouncement)

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM