可以将文章内容翻译成中文,广告屏蔽插件可能会导致该功能失效(如失效,请关闭广告屏蔽插件后再试):
问题:
So far I've been hacking together examples to get a feel for the APIs the Android SDK as a whole. However, I've hit an impasse.
I'm trying to Inflate a LinearLayout with 2 TextViews from an XML file. These will be then Views that are paged. I simply can't get this to work and realize that I totally don't understand what's going on with the code.
Looking at the source I can see that the method ViewPager.addNewItem calls the InstantiateItem from the supplied adapter. And addNewItem is called in populate()
. populate() is called in a number of other places.
Anyway, in the example, I've experienced that when the method is overridden for PagerAdapter one must include a call to addView()
on the ViewPager collection that is passed from the ViewPager
as an argument.
If I wish to add multiple views I thought this would be a case of calling addView() more than once but as instantiateItem() returns an Object to ViewPager (in the example I had it returns the view that it added) I don't know what to return. I've tried returning the inflated view and a number of other things.
Please, can someone explain what is happening here?
Much appreciated.
@Override public Object instantiateItem(View collection, int position) { layout = inflater.inflate(R.layout.animallayout, null); TextView tv = (TextView) layout.findViewById(R.id.textView1); tv.setText("1________________>" + position); TextView tv2 = (TextView) layout.findViewById(R.id.textView2); tv.setText("2________________>" + position); ((ViewPager) collection).addView(layout); return layout; }
回答1:
I've recently implemented this and this is my instantiateItem
method (made it a bit minimal for readability.
@Override public Object instantiateItem(View collection, int position) { Evaluation evaluation = evaluations.get(position); View layout = inflater.inflate(R.layout.layout_evaluation, null); TextView evaluationSummary = (TextView) layout.findViewById(R.id.evaluation_summary); evaluationSummary.setText(evaluation.getEvaluationSummary()); ((ViewPager) collection).addView(layout); return layout; } @Override public void destroyItem(View collection, int position, Object view) { ((ViewPager) collection).removeView((View) view); } @Override public boolean isViewFromObject(View view, Object object) { return view == object; }
So for the page that is displayed, I get the data from my evaluations
list using the position as the index. Then inflate the Layout
which has the views I will add my data too. Then I get the TextView
to set the evaluation summary text on. Then the whole Layout
is added to the ViewPager
. And finally the Layout
is also returned.
If you still can't get it post your code.
回答2:
I have the same misunderstanding how this adapter works and I have made some investigation.
The key particularity is that your views are stored in two places:
1 in ViewPager you add them by calling ((ViewPager) container).addView(view);
(here you need to store only three views, what you see and left and right neighborhoods)
2 in private final ArrayList mItems = new ArrayList();
this is member of ViewPager there are stored views with information about they possition (they are added here by calling adapter method mAdapter.instantiateItem(this, position);
)
static class ItemInfo { Object object; int position; boolean scrolling; float widthFactor; float offset; }
When you slides ViewPager
gets view position from mItems
array or instantiates it and compare this view with children of ViewPager
with adapters method public boolean isViewFromObject(View view, Object object)
. View which is equals to object
is displayed to user on ViewPager
. If there is no view then the blank screen is displayed.
Here is ViewPager
method where view is compared to object:
ItemInfo infoForChild(View child) { for (int i=0; i
If views position from mItems is not in range {currentposition -1,currentposition +1} then it will be destroyed:
mItems.remove(itemIndex); mAdapter.destroyItem(this, pos, ii.object);
the view of ViewPagers memory 1
It is one trap with destroyItem
when you slides forward firs is called destroyItem
and then new item is added, but when you slides backward first is added new item and then old is destroyed. If you try to use only three cached views you can get IllegalStateException: The specified child already has a parent.
while sliding backwards.