একটি ডাটাবেস সঙ্গে পুনর্ব্যবহারযোগ্য ব্যবহার


127

বর্তমানে পুনর্ব্যবহারযোগ্য ভিউ.এডাপ্টারের কোনও ডিফল্ট বাস্তবায়ন নেই।

অফিসিয়াল রিলিজের সাথে থাকতে পারে, গুগল এটি যুক্ত করবে।

যেহেতু জন্য কোন ধরনের সমর্থন নেই CursorAdapterসঙ্গে RecyclerViewবর্তমানে, কিভাবে আমরা একটি ব্যবহার করতে পারেন RecyclerViewএকটি ডাটাবেস সাথে? কোনও পরামর্শ ?

উত্তর:


101

আপনি যদি একটি দিয়ে কোয়েরি চালাচ্ছেন CursorLoaderএবং আপনি তার RecyclerViewপরিবর্তে চান ListView

আপনি আমার চেষ্টা করতে পারেন CursorRecyclerViewAdapter: রিসাইক্লারভিউতে কার্সারএডাপ্টার


12
কোড মুছে ফেলা এবং সন্নিবেশ করানোর পরে অ্যানিমেশনগুলিতে কাজ করে না কারণ কার্সার ক্লাসের পদ্ধতিটি রেজিস্ট্রেশনডেটসেটবার্সারটি সনাক্ত করতে পারে না কোন উপাদানটি ঠিক পরিবর্তন হয়েছে। সুতরাং প্রতিবার সামগ্রী সামগ্রী সরবরাহকারী পুনর্ব্যবহারযোগ্য দর্শনটি notifyDataSetChanged- এর মাধ্যমে সম্পূর্ণ লোড হয়ে যায় এবং এটিই রিসাইক্লার ভিউয়ের মূল অংশ।
ফ্রেঙ্কাস

1
প্রাসঙ্গিক ডেটা পেতে, কেবল কনস্ট্রাক্টরের মাধ্যমে অ্যাডাপ্টারে কার্সারটি প্রেরণ করে এবং কেবলমাত্র অনারবাইভহোল্ডার () -র মধ্যে कर्सर.মোভটোপজিশন () ব্যবহার করে কী ক্ষতি হবে?
গৌতম

2
পরামর্শ: কন্সট্রাক্টরের কাছ থেকে অদলবদল করুন () কল করুন, সুতরাং আপনাকে সেখানে নিজেকে পুনরাবৃত্তি করতে হবে না
গ্রেগ এনিস

2
@francas অ্যানিমেশন কাজ জরিমানা যদি আপনি সেটsetHasStableIds(true)
alders

2
@ আল্ডার্স কি আপনি সেটহ্যাসটেবলআইডস ছাড়াও অন্য কোনও পরিবর্তন করেছেন? আমি চেষ্টা করেছি যে পাশাপাশি, আইডি এর সাথেও মেলে না এবং যতবার আমি কার্সারটি স্যুপ করি তারপরে তালিকাটি শীর্ষে চলে যায়।
dwbrito

89

আমার সমাধানটি ছিল আমার রিসাইক্লারভিউ.এডাপ্টার বাস্তবায়নে একজন কার্সর অ্যাডাপ্টার সদস্য রাখা। তারপরে নতুন দর্শন তৈরির সমস্ত হ্যান্ডলিং পাস করা এবং এটি কার্সার অ্যাডাপ্টারের সাথে আবদ্ধ করা, এরকম কিছু:

public class MyRecyclerAdapter extends Adapter<MyRecyclerAdapter.ViewHolder> {

    // Because RecyclerView.Adapter in its current form doesn't natively 
    // support cursors, we wrap a CursorAdapter that will do all the job
    // for us.
    CursorAdapter mCursorAdapter;

    Context mContext;

    public MyRecyclerAdapter(Context context, Cursor c) {

        mContext = context;

        mCursorAdapter = new CursorAdapter(mContext, c, 0) {

            @Override
            public View newView(Context context, Cursor cursor, ViewGroup parent) {
                // Inflate the view here
            }

            @Override
            public void bindView(View view, Context context, Cursor cursor) {
                // Binding operations
            }
        };
    }

    public static class ViewHolder extends RecyclerView.ViewHolder {
        View v1;

        public ViewHolder(View itemView) {
            super(itemView);
            v1 = itemView.findViewById(R.id.v1);
        }
    }

    @Override
    public int getItemCount() {
        return mCursorAdapter.getCount();
    }

    @Override
    public void onBindViewHolder(ViewHolder holder, int position) {
        // Passing the binding operation to cursor loader
        mCursorAdapter.getCursor().moveToPosition(position); //EDITED: added this line as suggested in the comments below, thanks :)
        mCursorAdapter.bindView(holder.itemView, mContext, mCursorAdapter.getCursor());

    }

    @Override
    public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        // Passing the inflater job to the cursor-adapter
        View v = mCursorAdapter.newView(mContext, mCursorAdapter.getCursor(), parent);
        return new ViewHolder(v);
    }
}

17
এই সমাধানটির জন্য ধন্যবাদ, এটি আমার পক্ষে কাজ করে, আমাকে কেবলমাত্র বিনডউইউহোল্ডারটিতে একটি লাইন যুক্ত করতে হয়েছিল: 'mCursorAdapter.getCursor ()। পদচারণা অবস্থান (অবস্থান)' বিন্দু ভিউ () কল করার আগে mCursorAdapter
ইউজার 1071762

8
@nbtk কার্সারএডাপ্টারের পরিবর্তে কার্সার রাখবেন না কেন? আমি যেমন বুঝতে পেরেছি, আপনি কার্সরএডাপ্টারের কোনও দক্ষতা ব্যবহার করবেন না।
মাইডোগটম

14
@ মাইডগটম - আমি কোনও কার্সার ধরে রাখছি না, কারণ আপনার কাজটি শেষ হওয়ার সময় আপনাকে এটি সাফ করা দরকার, এবং এটি নূলে সেট করা যথেষ্ট নয়। কার্সরএডাপ্টার সঠিকভাবে এটি যত্ন করে care
nbtk

3
@nbtk আমি এই ধারণাটি সত্যিই পছন্দ করেছি এবং আমি তখন থেকেই এটি নিয়ে খেলছি। আমি এটিকে কিছুটা বিমূর্ত করার চেষ্টা করেছি এবং একটি লাইব্রেরি তৈরি করেছি, এটি 100% নয় যেখানে আমি এটি চাই এবং এটি কাজটি সম্পন্ন করে। আপনি এটি পরীক্ষা করে দেখতে আগ্রহী? আপনার যদি গিথুব থাকে তবে আমি আপনাকে অনুপ্রেরণা দিয়ে ক্রেডিট করতে খুশি হব। github.com/androidessence/RecyclerViewCursorAdapter
AdamMc331

5
@nbtk আপনার ডেটা নতুন ডেটা দিয়ে রিফ্রেশ করতে নিম্নলিখিতগুলি করুন:public void changeCursor(Cursor cursor){ mCursorAdapter.changeCursor(cursor); notifyDataSetChanged(); }
সিজো

48

যেহেতু আপনার প্রশ্নটি " RecyclerViewএকটি ডাটাবেস দিয়ে কীভাবে ব্যবহার করবেন" এবং আপনি এসকিউএলাইট বা এর সাথে অন্য কিছু চান কিনা তা সুনির্দিষ্ট করা হচ্ছে না RecyclerView, তাই আমি আপনাকে এমন একটি সমাধান দেব যা অত্যন্ত অনুকূল। আমি রাজ্যটিকে ডাটাবেস হিসাবে ব্যবহার করব এবং আপনাকে আপনার অভ্যন্তরের সমস্ত ডেটা প্রদর্শন করব RecyclerView। এটির ব্যবহার Loadersবা ব্যবহার ছাড়াই অ্যাসিঙ্ক্রোনাস ক্যোয়ারী সমর্থন রয়েছে AsyncTask

রাজত্ব কেন? realm.io পারফরম্যান্স অ্যান্ড্রয়েড

ধাপ 1

রিয়েলমের জন্য গ্রেড নির্ভরতা যুক্ত করুন, সর্বশেষ সংস্করণের জন্য নির্ভরতা এখানে পাওয়া যায়

ধাপ ২

উদাহরণস্বরূপ, আপনার মডেল বর্গ তৈরি করুন, এর মতো সহজ কিছু বলতে দিন Dataযার মধ্যে দুটি ক্ষেত্র রয়েছে, RecyclerViewসারিটির ভিতরে প্রদর্শিত একটি স্ট্রিং এবং একটি টাইমস্ট্যাম্প যা RecyclerViewআইটেম আইড হিসাবে আইটেম আইড হিসাবে ব্যবহৃত হবে আইটেমগুলিকে সঞ্চারিত করার অনুমতি দেবে । লক্ষ্য করুন যে আমি RealmObjectনীচে প্রসারিত করেছি যার কারণে আপনার Dataক্লাসটি একটি টেবিল হিসাবে সংরক্ষণ করা হবে এবং আপনার সমস্ত সম্পত্তি সেই টেবিলের কলাম হিসাবে সঞ্চয় করা হবে Data। আমি আমার ক্ষেত্রে ডেটা পাঠ্যটিকে প্রাথমিক কী হিসাবে চিহ্নিত করেছি যেহেতু আমি চাই না যে স্ট্রিংটি একাধিকবার যুক্ত করা হোক। তবে আপনি যদি সদৃশ থাকা পছন্দ করেন তবে @PrimaryKey হিসাবে টাইমস্ট্যাম্প তৈরি করুন। প্রাথমিক কী ছাড়া আপনার কোনও টেবিল থাকতে পারে তবে আপনি যদি সারিটি তৈরির পরে আপডেট করার চেষ্টা করেন তবে এটি সমস্যার সৃষ্টি করবে। এই উত্তরটি লেখার সময় একটি যৌগিক প্রাথমিক কী রিয়েল সমর্থন করে না।

import io.realm.RealmObject;
import io.realm.annotations.PrimaryKey;

public class Data extends RealmObject {
@PrimaryKey
private String data;

//The time when this item was added to the database
private long timestamp;

public String getData() {
    return data;
}

public void setData(String data) {
    this.data = data;
}

public long getTimestamp() {
    return timestamp;
}

public void setTimestamp(long timestamp) {
    this.timestamp = timestamp;
}
}

ধাপ 3

এর ভিতরে কীভাবে একটি একক সারি উপস্থিত হওয়া উচিত তার জন্য আপনার বিন্যাসটি তৈরি করুন RecyclerView। আমাদের ভিতরে একক সারি আইটেমের জন্য লেআউটটি Adapterনিম্নরূপ

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

<TextView
    android:id="@+id/area"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_gravity="center"
    android:background="@android:color/white"
    android:padding="16dp"
    android:text="Data"
    android:visibility="visible" />

</FrameLayout>

লক্ষ্য করুন যে আমি FrameLayoutএকটি TextViewঅভ্যন্তর থাকা সত্ত্বেও মূল হিসাবে রেখেছি । আমি এই বিন্যাসে আরও আইটেম যুক্ত করার পরিকল্পনা করছি এবং তাই এটি এখনই নমনীয় করে তুলেছে :)

সেখানকার কৌতূহলী লোকদের জন্য, বর্তমানে কোনও একক আইটেমটি দেখতে এটিই দেখা যায়। রিসাইক্লিউউয়ের ভিতরে একক আইটেম সারি বিন্যাস

পদক্ষেপ 4

আপনার RecyclerView.Adapterবাস্তবায়ন তৈরি করুন । এই ক্ষেত্রে, ডেটা উত্স অবজেক্টটি একটি বিশেষ অবজেক্ট বলা হয় RealmResultsযা মূলত একটি লাইভ ArrayList, অন্য কথায়, আইটেমগুলি আপনার টেবিল থেকে যুক্ত বা অপসারণ হওয়ায়, এই RealmResultsঅবজেক্টটি স্বয়ংক্রিয়ভাবে আপডেট হয়।

import android.content.Context;
import android.support.v7.widget.RecyclerView;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;

import io.realm.Realm;
import io.realm.RealmResults;
import slidenerd.vivz.realmrecycler.R;
import slidenerd.vivz.realmrecycler.model.Data;

public class DataAdapter extends RecyclerView.Adapter<DataAdapter.DataHolder> {
private LayoutInflater mInflater;
private Realm mRealm;
private RealmResults<Data> mResults;

public DataAdapter(Context context, Realm realm, RealmResults<Data> results) {
    mRealm = realm;
    mInflater = LayoutInflater.from(context);
    setResults(results);
}

public Data getItem(int position) {
    return mResults.get(position);
}

@Override
public DataHolder onCreateViewHolder(ViewGroup parent, int viewType) {
    View view = mInflater.inflate(R.layout.row_data, parent, false);
    DataHolder dataHolder = new DataHolder(view);
    return dataHolder;
}

@Override
public void onBindViewHolder(DataHolder holder, int position) {
    Data data = mResults.get(position);
    holder.setData(data.getData());
}

public void setResults(RealmResults<Data> results) {
    mResults = results;
    notifyDataSetChanged();
}

@Override
public long getItemId(int position) {
    return mResults.get(position).getTimestamp();
}

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

public void add(String text) {

    //Create a new object that contains the data we want to add
    Data data = new Data();
    data.setData(text);

    //Set the timestamp of creation of this object as the current time
    data.setTimestamp(System.currentTimeMillis());

    //Start a transaction
    mRealm.beginTransaction();

    //Copy or update the object if it already exists, update is possible only if your table has a primary key
    mRealm.copyToRealmOrUpdate(data);

    //Commit the transaction
    mRealm.commitTransaction();

    //Tell the Adapter to update what it shows.
    notifyDataSetChanged();
}

public void remove(int position) {

    //Start a transaction
    mRealm.beginTransaction();

    //Remove the item from the desired position
    mResults.remove(position);

    //Commit the transaction
    mRealm.commitTransaction();

    //Tell the Adapter to update what it shows
    notifyItemRemoved(position);
}

public static class DataHolder extends RecyclerView.ViewHolder {
    TextView area;

    public DataHolder(View itemView) {
        super(itemView);
        area = (TextView) itemView.findViewById(R.id.area);
    }

    public void setData(String text) {
        area.setText(text);
    }
}
}

লক্ষ্য করুন যে notifyItemRemovedঅপসারণটি ঘটেছিল সেই অবস্থানের সাথে আমি কল করছি তবে আমি কল করব না notifyItemInsertedবা notifyItemRangeChangedরিয়েলমের এন্ট্রিগুলি অর্ডার করা ফ্যাশনে সংরক্ষিত না হওয়ায় কোন পজিশনটি ডাটাবেসে সন্নিবেশ করা হয়েছে তা সরাসরি জানার উপায় নেই। RealmResultsঅবজেক্ট স্বয়ংক্রিয় যখনই একটি নতুন আইটেম, যোগ করা হয় পরিবর্তন করা বা ডাটাবেস থেকে মুছে তাই আমরা কল আপডেট notifyDataSetChangedযখন যুক্ত করা এবং ঢোকাতে বাল্ক এন্ট্রি। এই মুহুর্তে, আপনি সম্ভবত অ্যানিমেশনগুলি সম্পর্কে উদ্বিগ্ন যা ট্রিগার হবে না কারণ আপনি পদ্ধতির notifyDataSetChangedজায়গায় কল করছেন notifyXXX। ঠিক এই কারণেই আমার কাছে getItemIdপদ্ধতিটি ফলাফল অবজেক্ট থেকে প্রতিটি সারিটির টাইমস্ট্যাম্প ফেরত দেয়। notifyDataSetChangedযদি আপনি কল করেন setHasStableIds(true)এবং তারপরে ওভাররাইড করেন তবে এনিমেশন 2 টি পদক্ষেপে অর্জন করা হয়getItemId শুধু অবস্থান ব্যতীত অন্য কিছু সরবরাহ করা।

পদক্ষেপ 5

RecyclerViewআমাদের Activityবা যুক্ত করতে দেয় Fragment। আমার ক্ষেত্রে, আমি একটি ব্যবহার করছি Activity। এতে থাকা লেআউট ফাইলটি RecyclerViewবেশ তুচ্ছ এবং এটির মতো দেখতে হবে।

<android.support.v7.widget.RecyclerView xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@+id/recycler"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_margin="@dimen/text_margin"
app:layout_behavior="@string/appbar_scrolling_view_behavior" />

আমি একজন যোগ করেছেন app:layout_behaviorযেহেতু আমার RecyclerViewএকটি ভিতরে যায় CoordinatorLayoutযা আমি সংক্ষিপ্ততা জন্য এই উত্তর পোস্ট করা হয়নি।

পদক্ষেপ 6

আঁকো RecyclerViewকোডে এবং তথ্য দরকার সরবরাহ। ভিতরে একটি রাজ্য অবজেক্ট তৈরি করুন এবং সূচনা করুন এবং onCreateএকে একে বন্ধ করে দেওয়ার onDestroyমতো করে এটির অভ্যন্তরে বন্ধ করুন SQLiteOpenHelper। সবচেয়ে সহজতম আপনার onCreateভিতরে এর Activityচেহারা হবে। initUiপদ্ধতি যেখানে সব জাদু ঘটে থাকে। আমি ভিতরে রাজ্যের উদাহরণ খুলি onCreate

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    mRealm = Realm.getInstance(this);
    initUi();
}

private void initUi() {

    //Asynchronous query
    RealmResults<Data> mResults = mRealm.where(Data.class).findAllSortedAsync("data");

    //Tell me when the results are loaded so that I can tell my Adapter to update what it shows
    mResults.addChangeListener(new RealmChangeListener() {
        @Override
        public void onChange() {
            mAdapter.notifyDataSetChanged();
            Toast.makeText(ActivityMain.this, "onChange triggered", Toast.LENGTH_SHORT).show();
        }
    });
    mRecycler = (RecyclerView) findViewById(R.id.recycler);
    mRecycler.setLayoutManager(new LinearLayoutManager(this));
    mAdapter = new DataAdapter(this, mRealm, mResults);

    //Set the Adapter to use timestamp as the item id for each row from our database
    mAdapter.setHasStableIds(true);
    mRecycler.setAdapter(mAdapter);
}

লক্ষ করুন যে প্রথম পদক্ষেপে আমি Dataক্লাস থেকে সমস্ত বস্তুগুলিকে অ্যাসিঙ্ক্রোনাস পদ্ধতিতে ডেটা নামক ভেরিয়েবল নাম অনুসারে বাছাই করতে আমাকে জিজ্ঞাসা করি । এটি আমাকে RealmResultsপ্রধান থ্রেডে 0 টি আইটেম সহ একটি বস্তু দেয় যা আমি সেট করছি Adapter। আমি RealmChangeListenerযেখানে আমার সাথে কল করি সেই ব্যাকগ্রাউন্ড থ্রেড থেকে ডেটা লোড করা শেষ হয়ে গেলে আমি বিজ্ঞপ্তির জন্য একটি যুক্ত notifyDataSetChangedকরেছি Adapter। আমি প্রয়োগে যুক্ত হওয়া, সরানো বা সংশোধিত আইটেমগুলির ট্র্যাক রাখতে setHasStableIdsদেওয়া উচিত বলেও সত্য বলেছি RecyclerView.AdapteronDestroyআমার জন্য Activityপ্রচেষ্টা অধিরাজ্য উদাহরণস্বরূপ

@Override
protected void onDestroy() {
    super.onDestroy();
    mRealm.close();
}

এই পদ্ধতি initUiভিতরে বলা যেতে পারে onCreateআপনার এর Activityবা onCreateViewবা onViewCreatedআপনার এর Fragment। নিম্নলিখিত বিষয়গুলি লক্ষ্য করুন।

পদক্ষেপ 7

BAM! ভিতরে আপনার ডাটাবেস থেকে তথ্য নেই RecyclerViewasynchonously ছাড়া লোড CursorLoader, CursorAdapter, SQLiteOpenHelperঅ্যানিমেশনের সঙ্গে। এখানে প্রদর্শিত জিআইএফ চিত্রটি দারুণ লেগি তবে অ্যানিমেশনগুলি যখন আপনি আইটেমগুলি যুক্ত করবেন বা সেগুলি সরিয়ে ফেলছেন।

পুনর্ব্যবহারযোগ্য ভিউয়ের মধ্যে ডাটাবেস থেকে ডেটা


বিস্তৃত উদাহরণ প্রদানের জন্য ধন্যবাদ। আমার কয়েকটি প্রশ্ন আছে। আমি বাস্তবে বিটম্যাপ সঞ্চয় করতে চাই এটা কি সম্ভব? যদি হ্যাঁ, কলামের ধরণ কী হবে এবং কীভাবে এটি সংরক্ষণ এবং পুনরুদ্ধার করা যায়?
রকেশ

1
@ রাকেশ কখনই কোনও ডাটাবেসের মধ্যে চিত্র সংরক্ষণ করবেন না, সর্বদা এটি আপনার ফাইল সিস্টেমে সংরক্ষণ করুন এবং চিত্রের
ইউরি

দ্রুত সাড়া দেওয়ার জন্য ধন্যবাদ. আমি এমন একটি অ্যাপ তৈরি করছি যেখানে আমি আমার পরিচিতির জন্য চিত্রগুলি সঞ্চয় করতে চাই। আমি যদি ফাইল সিস্টেমে সংরক্ষণ করি তবে শত শত চিত্র থাকতে পারে। আমি কীভাবে ফাইল সিস্টেমে এই চিত্রগুলি পরিচালনা করব?
রাকেশ

9
সন্নিবেশে রিয়েলমের পারফরম্যান্সের একটি গ্রাফ সহ
এফডাব্লুআইডাব্লু

টুলবারটি দেওয়া বলে ভিভ্জ , এটি কি একটি @ স্লাইডেনার্ড সমাধান?
সিম্বলিক্সএইউ

1

আপনি প্রয়োজনীয় সমস্ত পদ্ধতি নিজেই প্রয়োগ করতে পারেন। আমি সম্প্রতি কার্সরএডাপ্টার থেকে কেবল পেস্টিং কোড অনুলিপি করে আমার নিজস্ব বাস্তবায়ন করেছি।

public class MyAdapter extends RecyclerView.Adapter<ViewHolder> {

        protected boolean mDataValid;
        protected boolean mAutoRequery;
        protected Cursor mCursor;
        protected Context mContext;
        protected int mRowIDColumn;
        protected ChangeObserver mChangeObserver;
        protected DataSetObserver mDataSetObserver;
        protected FilterQueryProvider mFilterQueryProvider;
        public static final int FLAG_AUTO_REQUERY = 0x01;
        public static final int FLAG_REGISTER_CONTENT_OBSERVER = 0x02;

        public Cursor getCursor() {
            return mCursor;
        }

        //Recommended
        public MyAdapter(Context context, Cursor c, int flags) {
            init(context, c, flags);
        }

        public MyAdapter(Context context, Cursor c) {
            init(context, c, FLAG_AUTO_REQUERY);
        }

        public MyAdapter(Context context, Cursor c, boolean autoRequery) {
            init(context, c, autoRequery ? FLAG_AUTO_REQUERY : FLAG_REGISTER_CONTENT_OBSERVER);
        }

        void init(Context context, Cursor c, int flags) {
            if ((flags & FLAG_AUTO_REQUERY) == FLAG_AUTO_REQUERY) {
                flags |= FLAG_REGISTER_CONTENT_OBSERVER;
                mAutoRequery = true;
            } else {
                mAutoRequery = false;
            }
            boolean cursorPresent = c != null;
            mCursor = c;
            mDataValid = cursorPresent;
            mContext = context;
            mRowIDColumn = cursorPresent ? c.getColumnIndexOrThrow("_id") : -1;
            if ((flags & FLAG_REGISTER_CONTENT_OBSERVER) == FLAG_REGISTER_CONTENT_OBSERVER) {
                mChangeObserver = new ChangeObserver();
                mDataSetObserver = new MyDataSetObserver();
            } else {
                mChangeObserver = null;
                mDataSetObserver = null;
            }

            if (cursorPresent) {
                if (mChangeObserver != null) c.registerContentObserver(mChangeObserver);
                if (mDataSetObserver != null) c.registerDataSetObserver(mDataSetObserver);
            }
        }

        // Create new views (invoked by the layout manager)
        @Override
        public ViewHolder onCreateViewHolder(final ViewGroup parent,
                                             int viewType) {
            // create a new view
            final View view = LayoutInflater.from(parent.getContext())
                    .inflate(R.layout.list_item, parent, false);
            // set the view's size, margins, paddings and layout parameters

            ViewHolder vh = new ViewHolder(view, mCursor, new ViewHolder.IMyViewHolderClicks() {

                @SuppressLint("NewApi")
                @Override
                public void onClick(Cursor cursor) {
                    Log.e("Item :", cursor.getString(cursor.getColumnIndex(MyDatabaseHelper.MW_NAAM)));
                    Intent intent = new Intent(TasksListFragment.this.getActivity(), DetailActivity.class);
                    intent.putExtra(DetailActivity.EXTRA_PARAM_ID, cursor.getLong(cursor.getColumnIndex(MyDatabaseHelper.MW_ID)));

                    ActivityOptions activityOptions = ActivityOptions.makeSceneTransitionAnimation(
                            TasksListFragment.this.getActivity(),

                            // Now we provide a list of Pair items which contain the view we can transitioning
                            // from, and the name of the view it is transitioning to, in the launched activity
                            new Pair<View, String>(
                                    view.findViewById(R.id.imageview_item),
                                    DetailActivity.VIEW_NAME_HEADER_IMAGE),
                            new Pair<View, String>(
                                    view.findViewById(R.id.textview_name),
                                    DetailActivity.VIEW_NAME_HEADER_TITLE)
                    );

                    // Now we can start the Activity, providing the activity options as a bundle
                    startActivity(intent, activityOptions.toBundle());
                    // END_INCLUDE(start_activity)
                }
            });
            return vh;
        }

        // Replace the contents of a view (invoked by the layout manager)
        @SuppressLint("NewApi")
        @Override
        public void onBindViewHolder(ViewHolder holder, int position) {
            // - get element from your dataset at this position
            // - replace the contents of the view with that element
            final Cursor cursor = getItem(position);

            holder.mTextView.setText(cursor.getString(cursor.getColumnIndex(MyDatabaseHelper.MW_NAAM)));
            holder.mImageView.setTransitionName("grid:image:" + cursor.getLong(cursor.getColumnIndex(MyDatabaseHelper.MW_ID)));
            holder.mTextView.setTransitionName("grid:name:" + cursor.getLong(cursor.getColumnIndex(MyDatabaseHelper.MW_ID)));
        }

        //@Override
        // public View getView(int position, View view, ViewGroup viewGroup) {
        //     return view;
        // }

        // Return the size of your dataset (invoked by the layout manager)
        @Override
        public int getItemCount() {
            return getCount();
        }

        public int getCount() {
            if (mDataValid && mCursor != null) {
                return mCursor.getCount();
            } else {
                return 0;
            }
        }

        public Cursor getItem(int position) {
            if (mDataValid && mCursor != null) {
                mCursor.moveToPosition(position);
                return mCursor;
            } else {
                return null;
            }
        }

        @Override
        public long getItemId(int position) {
            if (mDataValid && mCursor != null) {
                if (mCursor.moveToPosition(position)) {
                    return mCursor.getLong(mRowIDColumn);
                } else {
                    return 0;
                }
            } else {
                return 0;
            }
        }

        public Cursor swapCursor(Cursor newCursor) {
            if (newCursor == mCursor) {
                return null;
            }
            Cursor oldCursor = mCursor;
            if (oldCursor != null) {
                if (mChangeObserver != null) oldCursor.unregisterContentObserver(mChangeObserver);
                if (mDataSetObserver != null) oldCursor.unregisterDataSetObserver(mDataSetObserver);
            }
            mCursor = newCursor;
            if (newCursor != null) {
                if (mChangeObserver != null) newCursor.registerContentObserver(mChangeObserver);
                if (mDataSetObserver != null) newCursor.registerDataSetObserver(mDataSetObserver);
                mRowIDColumn = newCursor.getColumnIndexOrThrow("_id");
                mDataValid = true;
                // notify the observers about the new cursor
                notifyDataSetChanged();
            } else {
                mRowIDColumn = -1;
                mDataValid = false;
                // notify the observers about the lack of a data set
                notifyDataSetInvalidated();
            }
            return oldCursor;
        }

        public void changeCursor(Cursor cursor) {
            Cursor old = swapCursor(cursor);
            if (old != null) {
                old.close();
            }
        }

        public CharSequence convertToString(Cursor cursor) {
            return cursor == null ? "" : cursor.toString();
        }

        public Cursor runQueryOnBackgroundThread(CharSequence constraint) {
            if (mFilterQueryProvider != null) {
                return mFilterQueryProvider.runQuery(constraint);
            }
            return mCursor;
        }


        public FilterQueryProvider getFilterQueryProvider() {
            return mFilterQueryProvider;
        }

        public void setFilterQueryProvider(FilterQueryProvider filterQueryProvider) {
            mFilterQueryProvider = filterQueryProvider;
        }

        protected void onContentChanged() {
            if (mAutoRequery && mCursor != null && !mCursor.isClosed()) {
                if (false) Log.v("Cursor", "Auto requerying " + mCursor + " due to update");
                mDataValid = mCursor.requery();
            }
        }

        private class ChangeObserver extends ContentObserver {
            public ChangeObserver() {
                super(new Handler());
            }

            @Override
            public boolean deliverSelfNotifications() {
                return true;
            }

            @Override
            public void onChange(boolean selfChange) {
                onContentChanged();
            }
        }

        private class MyDataSetObserver extends DataSetObserver {
            @Override
            public void onChanged() {
                mDataValid = true;
                notifyDataSetChanged();
            }

            @Override
            public void onInvalidated() {
                mDataValid = false;
                notifyDataSetInvalidated();
            }
        }


        private final DataSetObservable mDataSetObservable = new DataSetObservable();

        public void registerDataSetObserver(DataSetObserver observer) {
            mDataSetObservable.registerObserver(observer);
        }

        public void unregisterDataSetObserver(DataSetObserver observer) {
            mDataSetObservable.unregisterObserver(observer);
        }

        public void notifyDataSetInvalidated() {
            mDataSetObservable.notifyInvalidated();
        }
    }

তাহলে এই ক্লাসে লোডার কোথায়?
ইগোরগানাপলস্কি

লোডার এই অ্যাডাপ্টারটি ব্যবহার করবে। দেখুন developer.android.com/guide/components/loaders.html কিভাবে লোডার ব্যবহার করতে একটি উদাহরণ জন্য। এই উদাহরণে সিম্পলসার্সার অ্যাডাপ্টারটি মাইএডাপ্টার হবে
আরনো

সিম্পলসার্সার অ্যাডাপ্টার বলতে কী বোঝ? আপনি রিসাইক্লারভিউ.এডাপ্টার প্রসারিত করছেন।
ইগোরগানাপলস্কি

1
আমি বোঝাতে চাইছি যে আপনি যদি উদাহরণটিতে সিম্পল কার্সার অ্যাডাপ্টারটি দেখেন তবে আপনি তার জায়গায় মাইএডাপ্টার লাগাতে পারেন। আপনি যে অ্যাডাপ্টারটি প্রসারিত করবেন ততক্ষণ তা বিবেচনা করে না যতক্ষণ না এতে স্যুপ্পকার্সার পদ্ধতি এবং অন্যান্য কিছু পদ্ধতি রয়েছে।
আরনো

উপরে উল্লিখিত @nbtk এর উত্তরের সাথে তুলনা করা হলে কিছুটা ওভার কোডিং মনে হচ্ছে।
eRaisedToX


1

অন্য একটি উত্তর, যেহেতু আমি স্বীকৃত উত্তরটি পছন্দ করি না (যা ইমোতে কোনও স্বজ্ঞাত ব্যবহার নেই)।

নিম্নলিখিতটি আমার নিজস্ব বাস্তবায়ন, যা এর সাথে খুব মিল (এবং আংশিকভাবে অনুপ্রাণিত হয়েছে) SimpleCursorAdapter:

public class RecyclerViewSimpleCursorAdapter extends RecyclerView.Adapter {
    private int mLayout;
    private Cursor mCursor;
    private String[] mFrom;
    private int[] mTo;

    private boolean mAutoRequery;
    private ContentObserver mContentObserver;

    /**
     * Standard constructor.
     *
     * @param layout resource identifier of a layout file that defines the views for this list item. The layout file should include at least those named views defined in "to"
     * @param c      The database cursor. Can be null if the cursor is not available yet.
     * @param from   A list of column names representing the data to bind to the UI. Can be null if the cursor is not available yet.
     * @param to     The views that should display column in the "from" parameter. These should all be TextViews and ImageViews. The first N views in this list are given the values of the first N columns in the from parameter. Can be null if the cursor is not available yet.
     */
    public RecyclerViewSimpleCursorAdapter(int layout, Cursor c, String[] from, int[] to, boolean autoRequery) {
        mLayout = layout;
        mCursor = c;
        mFrom = from;
        mTo = to;
        mAutoRequery = autoRequery;

        if (mAutoRequery) {
            initializeContentObserver();
        }
    }

    @Override
    public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        return new RecyclerView.ViewHolder(
                LayoutInflater.from(parent.getContext())
                        .inflate(mLayout, parent, false)
        ) {
        };
    }

    @Override
    public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
        mCursor.moveToPosition(position);

        if (mFrom == null || mTo == null)
            return;

        for (int i = 0; i < mFrom.length && i < mTo.length; i++) {
            String from = mFrom[i];
            int columnIndex = mCursor.getColumnIndex(from);
            String value = mCursor.getString(columnIndex);
            View view = holder.itemView.findViewById(mTo[i]);

            if (view instanceof TextView) {
                ((TextView) view).setText(value);
            } else if (view instanceof ImageView) {
                try {
                    ((ImageView) view).setImageResource(Integer.parseInt(value));
                } catch (NumberFormatException nfe) {
                    ((ImageView) view).setImageURI(Uri.parse(value));
                }
            } else {
                throw new IllegalStateException(view.getClass().getName() + " is not a view that can be bound by this RecyclerViewSimpleCursorAdapter");
            }
        }
    }

    @Override
    public int getItemCount() {
        return mCursor  != null ? mCursor.getCount() : 0;
    }

    private void initializeContentObserver() {
        mContentObserver = new ContentObserver(new Handler()) {
            @Override
            public boolean deliverSelfNotifications() {
                return true;
            }

            @Override
            public void onChange(boolean selfChange) {
                notifyDataSetChanged();
            }
        };
        mCursor.registerContentObserver(mContentObserver);
    }

    /**
     * Change the underlying cursor to a new cursor. If there is an existing cursor it will be closed.
     *
     * @param cursor The new cursor to be used
     */
    public void changeCursor(Cursor cursor) {
        Cursor oldCursor = mCursor;
        if (mAutoRequery) {
            if (mCursor != null) {
                mCursor.unregisterContentObserver(mContentObserver);
            }

            mContentObserver = new ContentObserver(new Handler()) {
                @Override
                public boolean deliverSelfNotifications() {
                    return true;
                }

                @Override
                public void onChange(boolean selfChange) {
                    notifyDataSetChanged();
                }
            };

            mCursor = cursor;
            if (mCursor != null) {
                mCursor.registerContentObserver(mContentObserver);
            }
        }

        notifyDataSetChanged();

        if (oldCursor != null && oldCursor != mCursor) {
            oldCursor.close();
        }
    }

    /**
     * Change the cursor and change the column-to-view mappings at the same time.
     *
     * @param cursor The database cursor. Can be null if the cursor is not available yet.
     * @param from A list of column names representing the data to bind to the UI. Can be null if the cursor is not available yet.
     * @param to The views that should display column in the "from" parameter. These should all be TextViews or ImageViews. The first N views in this list are given the values of the first N columns in the from parameter. Can be null if the cursor is not available yet.
     */
    public void changeCursorAndColumns(Cursor cursor, String[] from, int[] to) {
        mFrom = from;
        mTo = to;
        changeCursor(cursor);
    }

    /**
     * Returns the cursor.
     * @return the cursor
     */
    public Cursor getCursor() {
        return mCursor;
    }
}

আপনি অন্যান্য নির্দিষ্ট ব্যবহারের জন্য এটি পরিবর্তন করতে পারেন, তবে একটি কার্সার থাকা এটি যেমন কাজ করে ঠিক তেমন SimpleCursorAdapterএকটি দিয়ে RecyclerView


0

একটি তালিকায় ডিবি এবং স্টোর থেকে ডেটা পুনরুদ্ধার করার পরে, এটি এমন কিছু হওয়া উচিত:

    dbHelper = new BooksDbAdapter(this);
    dbHelper.open();
    //Clean all data
    dbHelper.deleteAllZist();
    //Add some data
    dbHelper.insertSomeRecords();
    List<String> mylist = dbHelper.getArrayColumn(3);

রিসাইক্লারভিউতে ডেটা যুক্ত করুন

list = new ArrayList<DataObject>();
 Integer i=0;
    for (String lst : mylist) {
        list.add(i++, new DataObject(lst,
                "The RecyclerView widget is a more advanced and flexible 
               version of ListView."));
    }

    recyclerView = (RecyclerView) findViewById(R.id.recyclerview);
    recyclerView.setHasFixedSize(true);
    myRecAdapter = new RecyclerviewAdapter(list, Zist1ChapterActivity.this);

0

নীচে আমার একটি পুনর্ব্যবহারের জন্য কার্সরডাপ্টারের প্রয়োগ রয়েছে। এটি অনআইটেমিক্লিকলিস্টনার, অনলং আইটেম ক্লিকলিস্টনার, অনফুটারক্লিকলিস্টনার, বিভাগ এবং পাদচরণ সমর্থন করে। সমর্থন করে না , শিরোলেখ, onHeaderClickListner, দ্রুত scroller, চটচটে হেডার বা চটচটে বিভাগে।

এই অ্যাডাপ্টারটি কেবল প্রসারিত করুন এবং আপনার নিজস্ব অ্যাডাপ্টার তৈরি করুন। প্রদত্ত পদ্ধতিগুলি ওভাররাইড করুন। এবং কার্সারটি অনকার্সরলডফিনিশড পদ্ধতি থেকে পাস করুন if যদি অ্যাডাপ্টারটি ইতিমধ্যে তৈরি হয়ে থাকে তবে অদলবদল ()

package com.tracker.paisa;

import android.database.Cursor;
import android.support.v7.widget.GridLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.util.Log;
import android.util.SparseBooleanArray;
import android.view.ViewGroup;
import java.util.ArrayList;
import java.util.List;


/**
 * @author Rahul Upadhyay (https://github.com/devDroidRaul)
 * Supports footer
 * Onitemclicklistener, OnItemLongClickListener, OnFooterClickListener
 * Supports Sections.
 *
 * Does Not support,Header, OnHeaderClickListener, FastScroller, StickySection (this can b done with item decor)
 * Pull requests are welcome for improvements.
 *
 * Override this to give logic to place subheaders between items.
 * public abstract boolean onPlaceSubheaderBetweenItems(int position);
 *
 * create seperate viewHolders for item, subheaders and footer. and return required views.
 *
 * @Override below methods for individual item type.
 * public abstract VH onCreateItemViewHolder(ViewGroup parent, int viewType);
 * public abstract SH onCreateSubheaderViewHolder(ViewGroup parent, int viewType);
 * public abstract FH onCreateFooterViewHolder(ViewGroup parent, int viewType);
 *
 * Bind your views with data here.
 * @Override below methods to bind data to individual item types.
 * public abstract void onBindSubHeaderViewHolder(SH holder, Cursor cursor);
 * public abstract void onBindItemViewHolder(VH holder, Cursor cursor);
 * public abstract void onBindFooterViewHolder(FH holder, int itemPosition);
 *
 * Item type -1,-2,-3 are reserved, kindly do not pass them in getItemViewType.
 */

public abstract class RecyclerViewCursorAdapter<SH extends RecyclerView.ViewHolder, VH extends RecyclerView.ViewHolder, FH extends RecyclerView.ViewHolder>
        extends RecyclerView.Adapter<RecyclerView.ViewHolder> {

    public static final String TAG = RecyclerViewCursorAdapter.class.getSimpleName();

    private static final int TYPE_SECTION_HEADER = -1;

    private static final int TYPE_MAIN_HEADER = -2;
    private static final int TYPE_FOOTER = -3;

   // private int headerLayout=0,viewLayout=0;
    boolean createHeader;

    private List<Integer> subheaderPositions = new ArrayList<>();

    private Cursor mCursor;
    private boolean mDataValid,footerRequired=false;
    private int mRowIDColumn;
    private SparseBooleanArray mSelectedItemsIds;



   // public RecyclerViewCursorAdapter() { }

    //constructor
    public RecyclerViewCursorAdapter(Cursor c,boolean footerRequired) {
        setHasStableIds(true);
        swapCursor(c);
        this.footerRequired = footerRequired;

        this.mSelectedItemsIds = new SparseBooleanArray();
    }



    // interface for listning click on recycler view;
    public interface OnItemClickedListener{
        void OnItemClick(int id, Object data);

        void onItemLongClick(int id);
    }

    OnItemClickedListener onItemClickedListener;

    public void setOnItemClickedListener(OnItemClickedListener onItemClickedListener) {
        this.onItemClickedListener = onItemClickedListener;
    }

    public interface OnFooterClickedListener{
        void onFooterClick(Object data);
    }

    OnFooterClickedListener onFooterClickedListener;
    public void setOnFooterClickedListener( OnFooterClickedListener onFooterClickedListener){
        this.onFooterClickedListener = onFooterClickedListener;
    }

    public interface OnHeaderClickedListener{
        void onHeaderClick(Object data);
    }

    OnHeaderClickedListener onHeaderClickedListener;
    public void setOnHeaderClickedListener( OnHeaderClickedListener onHeaderClickedListener){
        this.onHeaderClickedListener = onHeaderClickedListener;
    }

    private void initSubheaderPositions() {
        subheaderPositions.clear();

        if(getItemSize() != 0) {
            //TODO:Handle This please.
            //subheaderPositions.add(0);
        } else {
            return;
        }

        for(int i = 1; i < getItemSize(); i++) {
            if(onPlaceSubheaderBetweenItems(i - 1)) {
                subheaderPositions.add(i + subheaderPositions.size());
            }
        }
    }

    @Override
    public void onAttachedToRecyclerView(RecyclerView recyclerView) {

            initSubheaderPositions();



    }

    /**
     * Called when adapter needs to know whether to place subheader between two neighboring
     * items.
     *
     * @return true if you want to place subheader between two neighboring
     * items.
     */
    public abstract boolean onPlaceSubheaderBetweenItems(int position);

    public abstract VH onCreateItemViewHolder(ViewGroup parent, int viewType);

    public abstract SH onCreateSubheaderViewHolder(ViewGroup parent, int viewType);

    public abstract FH onCreateFooterViewHolder(ViewGroup parent, int viewType);

    public abstract void onBindSubHeaderViewHolder(SH holder, Cursor cursor);

    public abstract void onBindItemViewHolder(VH holder, Cursor cursor);

    public abstract void onBindFooterViewHolder(FH holder, int itemPosition);






    public abstract int getItemSize();

    /**
     * Return the view type of the item at position for the purposes
     * of view recycling.
     * Don't return -1. It's reserved for subheader view type.
     */
    public int getViewType(int position) {
        return 0;
    }

    @Override
    public final int getItemViewType(int position) {

        if(isSubheaderOnPosition(position)) {
            return TYPE_SECTION_HEADER;
        } if(footerRequired && getCursor().getPosition()==(getCursor().getCount()-1)){
            return TYPE_FOOTER;
        }else {
            return getViewType(position);
        }
    }

    public boolean isFooterAdded(){
        return footerRequired;
    }
    @Override
    public final RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        Log.d("RVCA-OCVH","create viewholder");
        if(viewType == TYPE_SECTION_HEADER) {
            return onCreateSubheaderViewHolder(parent, viewType);
        } if(footerRequired&&viewType == TYPE_FOOTER){
            return onCreateFooterViewHolder(parent, viewType);
        }else {
            return onCreateItemViewHolder(parent, viewType);
        }
    }

    @SuppressWarnings("unchecked")
    @Override
    public final void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
        Log.d("RVCA-OBVH","bind viewholder");
        Log.d("RVCA-OBVH","subheader position:"+isSubheaderOnPosition(position));
        if(isSubheaderOnPosition(position)) {
            if (!mDataValid) {
                throw new IllegalStateException("Cannot bind viewholder when cursor is in invalid state.");
            }
            if (!mCursor.moveToPosition(getItemPositionForViewHolder(position))) {
                throw new IllegalStateException("Could not move cursor to position " + getItemPositionForViewHolder(position) + " when trying to bind viewholder");
            }

            onBindSubHeaderViewHolder((SH)holder, mCursor);
        }if(footerRequired && position==getItemCount()-1){
            Log.d("RVCA-OBVH","bind footerHolder");
            onBindFooterViewHolder((FH) holder,position);

        } else {
            if (!mDataValid) {
                throw new IllegalStateException("Cannot bind viewholder when cursor is in invalid state.");
            }
            if (!mCursor.moveToPosition(getItemPositionForViewHolder(position))) {
                throw new IllegalStateException("Could not move cursor to position " + getItemPositionForViewHolder(position) + " when trying to bind viewholder");
            }
            // if(!mCursor.isAfterLast()) {
            //   mCursor.moveToPosition(position);
            onBindItemViewHolder((VH)holder, mCursor);

        }
    }

    @Override
    public final int getItemCount() {
        return getItemSize() + subheaderPositions.size()+(footerRequired?1:0);
    }

    public void notifyDataChanged() {
        initSubheaderPositions();
        notifyDataSetChanged();
    }

    public void notifyItemInsertedAtPosition(int itemPosition) {
        if (itemPosition == 0) {
            if (getItemCount() == 1 || onPlaceSubheaderBetweenItems(itemPosition)) {
                subheaderPositions.add(0, 0);
                increaseSubheaderPositions(1, 2);
                notifyItemRangeInserted(0, 2);
            } else {
                increaseSubheaderPositions(1, 1);
                notifyItemInserted(1);
            }
        } else if (itemPosition == getItemSize() - 1) {
            if (onPlaceSubheaderBetweenItems(itemPosition - 1)) {
                subheaderPositions.add(getItemCount() - 1);
                notifyItemRangeInserted(getItemCount() - 1, 2);
            } else {
                notifyItemInserted(getItemPositionInRecyclerView(itemPosition));
            }
        } else {
            if (onPlaceSubheaderBetweenItems(itemPosition - 1) && onPlaceSubheaderBetweenItems(itemPosition)) {
                final int itemPositionInRv = getItemPositionInRecyclerView(itemPosition - 1);
                final int countOfSubheadersBeforePosition = getCountOfSubheadersBeforePosition(itemPositionInRv);
                subheaderPositions.add(countOfSubheadersBeforePosition, itemPositionInRv + 1);
                increaseSubheaderPositions(countOfSubheadersBeforePosition + 1, 2);
                notifyItemRangeInserted(itemPositionInRv + 1, 2);
            } else if (onPlaceSubheaderBetweenItems(itemPosition)){
                final int itemPositionInRv = getItemPositionInRecyclerView(itemPosition - 1);
                increaseSubheaderPositions(getCountOfSubheadersBeforePosition(itemPositionInRv), 1);
                notifyItemInserted(itemPositionInRv + 1);
            } else if (onPlaceSubheaderBetweenItems(itemPosition - 1)) {
                final int itemPositionInRv = getItemPositionInRecyclerView(itemPosition);
                increaseSubheaderPositions(getCountOfSubheadersBeforePosition(itemPositionInRv), 1);
                notifyItemInserted(itemPositionInRv);
            } else {
                final int itemPositionInRv = getItemPositionInRecyclerView(itemPosition);
                increaseSubheaderPositions(getCountOfSubheadersBeforePosition(itemPositionInRv), 1);
                notifyItemInserted(itemPositionInRv);
            }
        }
    }

    public void notifyItemChangedAtPosition(int itemPosition) {
        final int itemPositionInRv = getItemPositionInRecyclerView(itemPosition);
        notifyItemChanged(itemPositionInRv);
    }

    public void notifyItemRemovedAtPosition(int itemPosition) {

        final int itemPositionInRv = getItemPositionInRecyclerView(itemPosition);

        for (int i = 1; i < subheaderPositions.size(); i++) {
            final int subheaderPosition = subheaderPositions.get(i);
            if (subheaderPosition > itemPositionInRv) {
                final int previousSubheaderPosition = subheaderPositions.get(i - 1);
                if (subheaderPosition - previousSubheaderPosition == 2) {
                    subheaderPositions.remove(subheaderPositions.indexOf(previousSubheaderPosition));
                    decreaseSubheaderPositions(subheaderPositions.indexOf(subheaderPosition), 2);
                    notifyItemRangeRemoved(itemPositionInRv - 1, 2);
                } else {
                    decreaseSubheaderPositions(subheaderPositions.indexOf(subheaderPosition), 1);
                    notifyItemRemoved(itemPositionInRv);
                }
                return;
            }
        }

        final int lastSubheaderPosition = subheaderPositions.get(subheaderPositions.size() - 1);
        if (itemPositionInRv - lastSubheaderPosition == 1 && getItemCount() == itemPosition + subheaderPositions.size()) {
            subheaderPositions.remove(subheaderPositions.size() - 1);
            notifyItemRangeRemoved(itemPositionInRv - 1, 2);
        } else {
            notifyItemRemoved(itemPositionInRv);
        }
    }

    public void setGridLayoutManager(final GridLayoutManager gridLayoutManager) {
        gridLayoutManager.setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() {
            @Override
            public int getSpanSize(int position) {
                if(subheaderPositions.contains(position)) {
                    return gridLayoutManager.getSpanCount();
                } else {
                    return 1;
                }
            }
        });
    }

    public boolean isSubheaderOnPosition(int position) {
        return subheaderPositions.contains(position);
    }

    public int getCountOfSubheadersBeforePosition(int position) {
        int count = 0;
        for(int subheaderPosition : subheaderPositions) {
            if(subheaderPosition < position) {
                count++;
            }
        }
        return count;
    }

    public int getItemPositionInRecyclerView(int position) {
        int countOfItems = 0;
        for (int i = 1; i < subheaderPositions.size(); i++) {
            final int previousSubheaderPosition = subheaderPositions.get(i - 1);
            final int nextSubheaderPosition = subheaderPositions.get(i);
            countOfItems += nextSubheaderPosition - previousSubheaderPosition - 1;
            if (countOfItems > position) {
                return position + i;
            }
        }
        return position + subheaderPositions.size();
    }

    public int getItemPositionForViewHolder(int viewHolderPosition) {
        return viewHolderPosition - getCountOfSubheadersBeforePosition(viewHolderPosition);
    }

    private void decreaseSubheaderPositions(int startSubheaderPosition, int decreaseNum) {
        for (int i = startSubheaderPosition; i < subheaderPositions.size(); i++) {
            final int subheaderPosition = subheaderPositions.get(i);
            subheaderPositions.set(i, subheaderPosition - decreaseNum);
        }
    }

    private void increaseSubheaderPositions(int startSubheaderPosition, int increaseNum) {
        for (int i = startSubheaderPosition; i < subheaderPositions.size(); i++) {
            final int subheaderPosition = subheaderPositions.get(i);
            subheaderPositions.set(i, subheaderPosition + increaseNum);
        }
    }

    private List<Integer> getSubheaderPositions() {
        return subheaderPositions;
    }

    public int getSubheaderCount() {
        return subheaderPositions.size();
    }

    public void swapCursor(Cursor newCursor) {
        Log.d("RVCA-SC","swap cursor");
        if (newCursor == mCursor) {
            Log.d("RVCA-SC","same cursor doing nothing");
            return;
        }

        if (newCursor != null) {
            Log.d("RVCA-SC","swap cursor not null");
            mCursor = newCursor;
            mRowIDColumn = mCursor.getColumnIndexOrThrow("_id");
            mDataValid = true;

            // notify the observers about the new cursor
            notifyDataChanged();
        } else {
            Log.d("RVCA-SC","swap cursor null");
            notifyItemRangeRemoved(0, getItemCount());
            mCursor = null;
            mRowIDColumn = -1;
            mDataValid = false;
        }
    }

    public Cursor getCursor(){
        return mCursor ;
    }

    @Override
    public long getItemId(int position) {
        if (isSubheaderOnPosition(position))
            return position;
        else {
            int cursorPosition = getItemPositionForViewHolder(position);
            Cursor cursor = getCursor();
            if (hasOpenCursor() && cursor.moveToPosition(cursorPosition)) {
                return cursor.getLong(cursor.getColumnIndex("_id"));
            }
            return NO_CURSOR_POSITION;
        }
    }
    public static final int NO_CURSOR_POSITION = -99;

    protected boolean hasOpenCursor() {
        Cursor cursor = getCursor();
        if (cursor == null || cursor.isClosed()) {
            swapCursor(null);
            return false;
        }
        return true;
    }

    //Methods to do Selection

    public void toggleSelection(int position) {
        selectView(position, !mSelectedItemsIds.get(position));
    }


    //Remove selected selections
    public void removeSelection() {
        mSelectedItemsIds = new SparseBooleanArray();
        notifyDataSetChanged();
    }


    //Put or delete selected position into SparseBooleanArray
    public void selectView(int position, boolean value) {
        if (value)
            mSelectedItemsIds.put(position, value);
        else
            mSelectedItemsIds.delete(position);

        // notifyItemChangedAtPosition(position);
        notifyDataSetChanged();
    }

    //Get total selected count
    public int getSelectedCount() {
        return mSelectedItemsIds.size();
    }

    //Return all selected ids
    public SparseBooleanArray getSelectedIds() {
        return mSelectedItemsIds;
    }

    public void setSelectedItemIds(SparseBooleanArray selectedItemIds){
        this.mSelectedItemsIds=selectedItemIds;
    }


}

0

সহজ রূপায়ণ হ'ল অ্যাডাপ্টারে কার্সার অবজেক্টটি ব্যবহার করা এবং ভিউ পরিবর্তনকারী ও অনবিন্দ ভিউহোল্ডারের কার্সারের অবস্থান পরিবর্তন করার জন্য ভিউহোল্ডার কনস্ট্রাক্টরের কাছে কার্সার পাস করা।

আপনি যদি কার্সার লোডার ব্যবহার করছেন তবে সেট কর্সার পদ্ধতিটিতে কল করুন onLoadfinished()এবং নাল ইন করুনonLoadReset()

public class PetCursorRecyclerViewAdapter extends RecyclerView.Adapter<PetCursorRecyclerViewAdapter.ViewHolder> {
Cursor cursor = null;
Context context;

public PetCursorRecyclerViewAdapter(Context context) {
    this.context = context;

}

public void setCursor(Cursor cursor) {
    this.cursor = cursor;
    notifyDataSetChanged();
}

@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
    View v = LayoutInflater.from(context).inflate(R.layout.catalog_item, parent, false);
    return new ViewHolder(v);
}

@Override
public void onBindViewHolder(ViewHolder holder, int position) {
    this.cursor.moveToPosition(position);
    holder.bindModel(this.cursor);
}

/*   getItemCount() returns the count of videos from the Cursor, or 0 if the
   Cursor is null (mimicking the behavior of CursorAdapter, which also treats
   a null Cursor as merely being one that has no rows)*/
@Override
public int getItemCount() {
    if (cursor == null) {
        return 0;
    } else {
        return cursor.getCount();
    }
}

public static class ViewHolder extends RecyclerView.ViewHolder {
    private TextView name_tv, breed_tv;

    public ViewHolder(View itemView) {
        super(itemView);
        name_tv = (TextView) itemView.findViewById(R.id.name_tv);
        breed_tv = (TextView) itemView.findViewById(R.id.breed_tv);
    }

    public void bindModel(Cursor cursor) {
        int name_index = cursor.getColumnIndexOrThrow(PetsContract.PetEntry.COLUMN_PET_NAME);
        int breed_index = cursor.getColumnIndexOrThrow(PetsContract.PetEntry.COLUMN_PET_BREED);
        name_tv.setText(cursor.getString(name_index));
        String breed = cursor.getString(breed_index);
        if (TextUtils.isEmpty(breed)) {
            breed = "Unknown Breed";
        }
        breed_tv.setText(breed);
    }
}    

}


0

শেষ পর্যন্ত আমাদের ডাটাবেস / নেটওয়ার্কের জন্য রিসাইক্লার ভিউ অ্যাডাপ্টারের একটি প্রয়োগ রয়েছে।

এটি একটি অ্যান্ড্রয়েড আর্কিটেকচার উপাদানগুলির সাথে উপলব্ধি করেছে :

কক্ষের ডাটাবেস: এসকিউএল ডাটাবেসের শীর্ষে থাকা ডাটাবেস স্তর যা আপনি যেটির সাহায্যে হ্যান্ডল করার জন্য যে জাগতিক কাজগুলির যত্ন নেন SQLiteOpenHelper। অন্তর্নিহিত এসকিউএল ডাটাবেসটিতে অ্যাক্সেস পয়েন্ট হিসাবে পরিবেশন করা ডাটাবেসধারক। রুম ডাটাবেসটি এসকিউএল ডাটাবেসটিতে প্রশ্ন জারি করতে ডিএও ব্যবহার করে।

ভিউমোডেল: ইউআই-তে ডেটা সরবরাহ করে। সংগ্রহস্থল এবং UI এর মধ্যে একটি যোগাযোগ কেন্দ্র হিসাবে কাজ করে। ইউআই থেকে ডেটা উত্পন্ন যেখানে লুকায়। ভিউমোডেল দৃষ্টান্তগুলি ক্রিয়াকলাপ / টুকরো বিনোদনের জন্য বেঁচে থাকে।

লাইভডেটা: একটি ডেটা ধারক শ্রেণি যা পর্যবেক্ষণ করা যায়। সর্বদা ডেটার সর্বশেষ সংস্করণ ধারণ / ধারণ করে। ডেটা পরিবর্তন হয়ে গেলে তার পর্যবেক্ষককে অবহিত করে। লাইভডেটা সচেতন সচেতন। ইউআই উপাদানগুলি কেবল প্রাসঙ্গিক ডেটা পর্যবেক্ষণ করে এবং পর্যবেক্ষণটি থামায় বা পুনরায় শুরু করবে না। লাইভডেটা স্বয়ংক্রিয়ভাবে এগুলি সব পরিচালনা করে কারণ এটি পর্যবেক্ষণের সময় সম্পর্কিত লাইফসাইকেলের স্থিতি পরিবর্তনগুলি সম্পর্কে সচেতন।


-2

বিশদ সংরক্ষণের জন্য আপনি এসকিউএল ডাটাবেস ব্যবহার করতে পারেন। এটি ডেটা অ্যাক্সেস করা সহজ। আপনি গিথুবে আমার কোডটি চেক করতে পারেন। https://github.com/thiru-wta/ToDo

     database = new Database(this);
    getSupportActionBar().setTitle("To Do List");
    etAddItems = (EditText) findViewById(R.id.etAddItem);
    btnAdd = (Button) findViewById(R.id.btnAdd);
    mRecyclerView = (RecyclerView) findViewById(R.id.listView);
    mRecyclerView.setLayoutManager(new LinearLayoutManager(this));
    adapter = new RecyclerAdapter(this, listData);
    mRecyclerView.setAdapter(adapter);

    btnAdd.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            event_name = etAddItems.getText().toString();
            String getname="";

              database.storeEventDetails(event_name);

                 getname = database.getEventDetails(event_name);


            if (getname.length() != 0) {
                listData.add(getname);
                etAddItems.setText("");

                adapter.notifyData(listData);
            }

ডাটাবেস পদ্ধতি:

    public void storeEventDetails(String event_name, long timerStart) {
    SQLiteDatabase db1 = getWritableDatabase();
    db1.execSQL("insert into '"+event_details_tbl+"' values('" + event_name + "')");
    db1.close();
}

পদ্ধতি পান:

     public String getEventDetails(String event_name) {
    SQLiteDatabase db1 = getReadableDatabase();
    Cursor cur = db1.rawQuery("select * from '"+event_details_tbl+"' where     event_name ='" + event_name + "'", null);
    cur.moveToFirst();
    String evName = null;
    if (cur != null) {
        do {
            int eventName = cur.getColumnIndex("event_name");

            String ev_name = cur.getString(eventName);

            evName = ev_name;
        } while (cur.moveToNext());

    }
    cur.close();
    db1.close();
    return evName;

}     

-4

আপনার কেবল এই পদক্ষেপগুলি অনুসরণ করতে হবে:

  • রিসাইক্লারে view.adapterএকটি অ্যারেলিস্ট তৈরি করুন বা একটি শালীন মান দিয়ে শুরুর একটি আইট্রেটার তৈরি করুন।
  • উদাহরণস্বরূপ, প্রাক্তন সোয়াপকার্সারের মতো একটি পদ্ধতি তৈরি করুন swapdata(Arraylist<T> data)। ভিতরে আপনি অ্যারে তালিকায় একটি নতুন মান প্রদান করেন, পুনরাবৃত্তাকারী বা যে কোনও কাঠামো যা সিস্টেমটি বিন্ধ্যদৃষ্টির সাথে পূর্ণসংখ্যা অবস্থান ব্যবহার করে পুনরাবৃত্তি করতে পারে can এই পদ্ধতির মানগুলিতে পাস করা হয় onloaderfinished()। তারপরে দায়িত্ব অর্পণের ঠিক পরে ফোন করুন notifydatachange(); এগুলি RecyclerViewঅ্যারেলিস্টের নতুন ডেটা সহ সমস্ত তালিকা পুনরায় আঁকতে বলবে ।

যে ক্রিয়াকলাপ বা খণ্ডটিতে আপনি লোডারকালব্যাক ব্যবহার করছেন সেখানে অ্যাডাপ্টারে নির্বাচিত ডেটা কাঠামোর উপর নির্ভর করে একটি পদ্ধতি তৈরি করুন যা কার্সারকে অ্যারেলিস্টে বা পুনরুক্তিতে রূপান্তর করে।

এইভাবে আপনি সর্বদা মূল থ্রেডটি ব্লক করে লোডেরাক্যালব্যাকটি ব্যবহার করতে পারেন।

আমাদের সাইট ব্যবহার করে, আপনি স্বীকার করেছেন যে আপনি আমাদের কুকি নীতি এবং গোপনীয়তা নীতিটি পড়েছেন এবং বুঝতে পেরেছেন ।
Licensed under cc by-sa 3.0 with attribution required.