如何在 ViewPager 中使用 cursorLoader?

Posted

技术标签:

【中文标题】如何在 ViewPager 中使用 cursorLoader?【英文标题】:How can I use a cursorLoader in a ViewPager? 【发布时间】:2015-09-13 11:52:54 【问题描述】:
I have a `cursorLoader` which is working fine.

问题是我没有按预期使用它,我将cursorLoader 中的数据加载到arrayLists 中,然后使用列表。

我发现本教程展示了如何使用 cursorLoaderviewPager,但我不明白如何真正实现它。

http://tumble.mlcastle.net/post/25875136857/bridging-cursorloaders-and-viewpagers-on-android

我有一个如下所示的片段:

public class FirstFragment extends Fragment 
    MapView mapView;
    GoogleMap map;
    // Store instance variables
    private String email,about,imagepath,latitude,longitude;
    Button getDirections;

    // newInstance constructor for creating fragment with arguments
    public static FirstFragment newInstance(String email,String about,String imagepath, String latitude, String longitude) 
        FirstFragment fragmentFirst = new FirstFragment();
        Bundle args = new Bundle();
        args.putString("email", email);
        args.putString("about", about);
        args.putString("imagepath", imagepath);
        args.putString("latitude", latitude);
        args.putString("longitude", longitude);
        fragmentFirst.setArguments(args);
        return fragmentFirst;
    

    // Store instance variables based on arguments passed
    @Override
    public void onCreate(Bundle savedInstanceState) 
        super.onCreate(savedInstanceState);
        email = getArguments().getString("email");
        about = getArguments().getString("about");
        imagepath = getArguments().getString("imagepath");
        latitude = getArguments().getString("latitude");
        longitude = getArguments().getString("longitude");
    

    // Inflate the view for the fragment based on layout XML
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             final Bundle savedInstanceState) 
        View view = inflater.inflate(R.layout.zzfragment_pager_items, container, false);
        ImageView imageView = (ImageView) view.findViewById(R.id.listpager_imageView);
        TextView about = (TextView) view.findViewById(R.id.listpager_text);
        TextView emaill = (TextView) view.findViewById(R.id.listpager_title);
        about.setText(this.about);
        emaill.setText(this.email);

        ImageLoader imageLoader = ImageLoader.getInstance();
        DisplayImageOptions options = new DisplayImageOptions.Builder().cacheInMemory(true)
                .cacheOnDisc(true).resetViewBeforeLoading(true)
                .considerExifParams(true)
                .build();
        imageLoader.getInstance().displayImage(imagepath, imageView, options);

        getDirections = (Button) view.findViewById(R.id.getdirections);

        getDirections.setOnClickListener(new View.OnClickListener() 
            @Override
            public void onClick(View v) 

                String strUri = "http://maps.google.com/maps?q=loc:" + latitude + "," + longitude + " (" + email + ")";
                Intent mapIntent = new Intent(android.content.Intent.ACTION_VIEW, Uri.parse(strUri));

                mapIntent.setClassName("com.google.android.apps.maps", "com.google.android.maps.MapsActivity");

                getActivity().startActivity(mapIntent);

            
        );

//        View v = inflater.inflate(R.layout.listviewtopager, container, false);
        // Gets the MapView from the XML layout and creates it
        mapView = (MapView) view.findViewById(R.id.mapview);
        mapView.onCreate(savedInstanceState);

        // Gets to GoogleMap from the MapView and does initialization stuff
        map = mapView.getMap();
        map.getUiSettings().setMyLocationButtonEnabled(false);
        map.setMyLocationEnabled(true);

        // Needs to call MapsInitializer before doing any CameraUpdateFactory calls
        MapsInitializer.initialize(this.getActivity());

        // Updates the location and zoom of the MapView
        CameraUpdate cameraUpdate = CameraUpdateFactory.newLatLngZoom(new LatLng(Double.parseDouble(latitude), Double.parseDouble(longitude)), 10);
        map.animateCamera(cameraUpdate);

        return view;
    

    @Override
    public void onResume() 
        mapView.onResume();
        super.onResume();
    

    @Override
    public void onDestroy() 
        super.onDestroy();
        mapView.onDestroy();
    

    @Override
    public void onLowMemory() 
        super.onLowMemory();
        mapView.onLowMemory();
    

我在这堂课上这样称呼它:

public class ViewPagerFragment extends FragmentActivity implements
        LoaderManager.LoaderCallbacks<Cursor>

    ArrayList<String> e

mail = new ArrayList<String>();
ArrayList<String> about = new ArrayList<String>();

ArrayList<String> imagepath = new ArrayList<String>();

ArrayList<String> latitude = new ArrayList<String>();

ArrayList<String> longitude = new ArrayList<String>();

@Override
protected void onCreate(Bundle savedInstanceState) 
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_view_pager_fragment);
    getLoaderManager().initLoader(1, null, this);


private SmartFragmentStatePagerAdapter adapterViewPager;

// Extend from SmartFragmentStatePagerAdapter now instead for more dynamic ViewPager items
public static class MyPagerAdapter extends SmartFragmentStatePagerAdapter 
    private final ArrayList<String> email;
    private final ArrayList<String> about;
    private final ArrayList<String> imagepath;
    private final ArrayList<String> latitude;
    private final ArrayList<String> longitude;
    private int listPosition;

    public MyPagerAdapter(FragmentManager fragmentManager,ArrayList<String> email,ArrayList<String> about, ArrayList<String> imagepath,ArrayList<String> latitude,ArrayList<String> longitude,int lPosition) 
        super(fragmentManager);
        this.imagepath=imagepath;
        this.email=email;
        this.about = about;
        this.latitude= latitude;
        this.longitude = longitude;
        listPosition = lPosition;
    

// Returns total number of pages
@Override
public int getCount() 
    return email.size();




 // Returns the fragment to display for that page
    @Override
        public Fragment getItem(int position) 
//            return FirstFragment.newInstance(listPosition+position, email.get(listPosition+position));
            return FirstFragment.newInstance(email.get(position), about.get(position),imagepath.get(position),latitude.get(position),longitude.get(position));
    

    // Returns the page title for the top indicator
    @Override
    public CharSequence getPageTitle(int position) 
        return "Page " + position;
    



@Override
public Loader<Cursor> onCreateLoader(int id, Bundle args) 
    String[] projection =
            
                    ActivitiesTable.KEY_EMAIL,
                    ActivitiesTable.KEY_ABOUT,
                    ActivitiesTable.KEY_IMAGEPATH,
                    ActivitiesTable.KEY_LATITUDE,
                    ActivitiesTable.KEY_LONGITUTDE

            ;
    CursorLoader cursorLoader = new CursorLoader(this, NameContentProvider.NAME_ACTIVITIES_URI, projection,
            null, null, null);
    return cursorLoader;


@Override
public void onLoadFinished(android.content.Loader<Cursor> loader, Cursor cursor) 
    if (cursor != null && cursor.getCount() > 0) 
        cursor.moveToFirst();
        do 
            email.add(cursor.getString(cursor
                    .getColumnIndexOrThrow("email")));
            about.add(cursor.getString(cursor
                    .getColumnIndexOrThrow("about")));
            imagepath.add(cursor.getString(cursor
                    .getColumnIndexOrThrow("imagepath")));
            latitude.add(cursor.getString(cursor
                    .getColumnIndexOrThrow("serverLatitude")));
            longitude.add(cursor.getString(cursor
                    .getColumnIndexOrThrow("serverLongitude")));
         while (cursor.moveToNext());
    

    int listPosition = getIntent().getExtras().getInt("position");


    ViewPager vpPager = (ViewPager) findViewById(R.id.vpPager);
    adapterViewPager = new MyPagerAdapter(getSupportFragmentManager(),email,about,imagepath,latitude,longitude,listPosition);
    vpPager.setAdapter(adapterViewPager);
    vpPager.setCurrentItem(listPosition);



@Override
public void onLoaderReset(android.content.Loader<Cursor> loader) 



   

如何修改代码以通过使用该教程直接将 viewPager 与加载器一起使用,而不是将所有内容存储在列表中然后使用列表?

【问题讨论】:

首先,与其处理 5 个不同的 ArrayList,为什么不简单地创建一个代表所有 5 个值的类并将这些对象放入一个 ArrayList 中?这种方法会让你的代码更干净,对我来说,结果很好 我不想再使用这些列表了。我已经找到了一种直接从 cursorLoader 使用数据的方法,但它使用了一些不推荐使用的方法。这个看起来更好,但我不知道如何使它工作 我使用的方法直接从加载器加载数据而不使用数组列表扩展 pagerAdapter 并使用已弃用的 instantiateItemdestroyItem。我更喜欢我在这里发布的方法,但我无法让它发挥作用。 将 instanceiteItem 和 destroyItem 中的视图参数转换为 ViewGroup - 不推荐使用视图参数化方法,根据文档,您应该使用 ViewGroup 变体 做到了。谢谢你。我有很多将显示的视图。最好使用 FragmentStatePagerAdapter 而不是简单的 PagerAdapter ? 【参考方案1】:

我无法评论,所以我正在写一个答案..

您有一个实现LoaderCallbacks&lt;Cursor&gt; 的活动。加载数据时,您的活动会收到 onLoadFinished 回调。在此方法中,您有一个 Cursor,应该显示在您的 ViewPager 中。

要显示来自Cursor 的数据,请在适配器上调用swapCursor 方法。因此,不要在每次加载数据时都创建适配器。创建一次,然后调用swapCursor 就可以了。

另外,不要每次都找到ViewPager - findViewById 是一个繁重的操作,应该在创建视图层次结构后执行。

所以,您的onLoadFinished 将如下所示:

@Override
public void onLoadFinished(android.content.Loader<Cursor> loader, Cursor cursor) 
    if (adapterViewPager == null) 
        adapterViewPager = new MyPagerAdapter(getSupportFragmentManager(), cursor);
        vpPager.setAdapter(adapterViewPager);
     else 
        adapterViewPager.swapCursor(cursor);
    

【讨论】:

【参考方案2】:

如果我在你身边,我要做的第一件事是创建一个模型类,它将保存光标中包含的信息。

public class Information implements Parcelable 

   public String imagepath;
   public String email;
   public String about;
   public String latitude;
   public String longitude;


protected Information(Parcel in) 
    imagepath = in.readString();
    email = in.readString();
    about = in.readString();
    latitude = in.readString();
    longitude = in.readString();


@Override
public int describeContents() 
    return 0;


@Override
public void writeToParcel(Parcel dest, int flags) 
    dest.writeString(imagepath);
    dest.writeString(email);
    dest.writeString(about);
    dest.writeString(latitude);
    dest.writeString(longitude);


@SuppressWarnings("unused")
public static final Parcelable.Creator<Information> CREATOR = new Parcelable.Creator<Information>() 
    @Override
    public Information createFromParcel(Parcel in) 
        return new Information(in);
    

    @Override
    public Information[] newArray(int size) 
        return new Information[size];
    
;
 

这已经是大部分内容了。我要做的第二件事是创建两个辅助方法,一个创建此类的实例,从 Cursor 读取数据,另一个创建一个集合。

 public static Information createInfoFromCursor(Cursor c) 
         Information info = new Information();
         info.email = cursor.getString(cursor
                    .getColumnIndexOrThrow("email")));
         info.about = cursor.getString(cursor
                    .getColumnIndexOrThrow("about")));
         info.imagepath =(cursor.getString(cursor
                    .getColumnIndexOrThrow("imagepath")));
         info.latitude = (cursor.getString(cursor
                    .getColumnIndexOrThrow("serverLatitude")));
         info.longitude = (cursor.getString(cursor
                    .getColumnIndexOrThrow("serverLongitude")));
         return info;
   

 public static ArrayList<Information> createInfoListFromCursor(Cursor c) 
    ArrayList<Information> info = new ArrayList<>();
    while(c.moveToNext()) 
        info.add(createInfoFromCursor());
    
    return info;
 

现在您可以决定向适配器提供CursorArrayList&lt;Information&gt;。如果您决定通过Cursor,在您的适配器中您将拥有

Cursor cursor;

 public MyPagerAdapter(FragmentManager fragmentManager,Cursor c) 
    super(fragmentManager);
    cursor = c;



@Override
public Fragment getItem(int position) 
   cursor.moveAtPosition(position);
   Information info = createInfoFromCursor(cursor);
   // create a version of `newInstance` that takes an Information object
   return FirstFragment.newInstance(info);


@Override
public int getCount() 
  return cursor == null ? 0 : cursor.getCount();
      

public Cursor swapCursor(Cursor newCursor) 
    if (newCursor == cursor) 
        return null;
    
    cursor = newCursor;
    if (newCursor != null) 
        notifyDataSetChanged();
     
    return oldCursor;

在您的活动中,创建适配器的实例,传递一个空游标,并将其保留为成员类。当调用onLoadFinished 时,使用该引用调用swapCursor,以刷新Adapter 的数据集

【讨论】:

【参考方案3】:

只需将光标保留在视图寻呼机适配器中,并在需要时从光标加载数据,而不是保留 5 个数据数组。这样,如果您的光标中有很多项目,UI 就不会阻塞。

您还可以添加一种方法来交换光标并通知数据已更改,并且将刷新 viewpager,而不是每次都重新创建和附加适配器(正如您提到的链接中一样)

MyPagerAdapter 应该如下所示:

Cursor cursor;

// Returns total number of pages 
@Override 
public int getCount()  
    return cursor==null?0:cursor.getCount();
 

// Returns the fragment to display for that page 
@Override 
public Fragment getItem(int position) 
    // position the cursor first
    cursor.moveToPosition(position);

    // read each field
    String email = cursor.getString(cursor
                    .getColumnIndexOrThrow("email"));
    ...
    return FirstFragment.newInstance(email, ...);
 

public void swapCursor(Cursor cursor) 
    this.cursor = cursor;
    // notify that data has changed and viewpager needs to be refreshed
    notifyDataSetChanged();

装载机重置后别忘了清理

@Override 
public void onLoaderReset(android.content.Loader<Cursor> loader) 
    adapterViewPager.swapCursor(null) 

【讨论】:

【参考方案4】:

如果您正在寻找一个很好的简单示例,说明如何使用光标中的片段填充 ViewPager,请查看我为 FOSDEM Companion 应用程序编写的 this Activity 代码。

注意:您应该避免将地图放在 ViewPager 中,因为触摸事件会被 MapView 拦截,用户将无法滑动到下一页或上一页。

【讨论】:

以上是关于如何在 ViewPager 中使用 cursorLoader?的主要内容,如果未能解决你的问题,请参考以下文章

如何在 ViewPager 中使用 Fragments

如何在 viewPager 中使用按钮更改页面?

如何使用 tabLayout 在 ViewPager 中显示片段?

如何在 Android 中使用 TabLayout 和 ViewPager2

如何在Android中使用viewPagerindicator和ViewPager?

如何在ViewPager中渲染单个片段?