Im trying to Enlarge the Centre most listview item. (Eventually, I hope to expand this into a sort of simple and light 3D effect that elements will appear to zoom in when in
1st no text appearing
On your MySimpleArrayAdapter, getView method. You forgot to setText for your textView. There is why no text is showing.
2nd make middle item bigger
You try to getLastVisiblePosition before your listview adapter is setup. I guess value of v must be 0 and position == 0. There is why the topper item become bigger. If you try change it to position == 1 or 2, maybe the second or third item will become bigger.
In order to make middle item become bigger, setOnScrollLinstener to listview
public int currentLargedPosition = 0 // global value
listView.setOnScrollListener(new OnScrollListener(){
@Override
public void onScroll(AbsListView arg0, int firstVisibleItem, int visibleItemCount, int totalItemCount){
Log.i("Scroll","first "+firstVisibleItem+", visibleItemCount "+visibleItemCount+",totalCount "+totalItemCount);
int center = (visibleItemCount)/2 + firstVisibleItem;
if(currentLargedPosition != center){
enlargeMiddleView(currentLargedPosition-firstVisibleItem, center-firstVisibleItem);
currentLargedPosition = center;
}
}
@Override
public void onScrollStateChanged(AbsListView arg0, int arg1) {
Log.i("scroll","onScrollStateChanged");
}
});
void enlargeMiddleView(int oldPosition, int newPosition){
// get enlarged view and make it return default size
TextView newTextView = (TextView)listView.getChildAt(oldPosition).findViewById(R.id.text);
newTextView.setHeight(50);
// get the current center view and make it bigger
TextView oldTextView = (TextView)listView.getChildAt(newPosition).findViewById(R.id.text);
oldTextView.setHeight(300);
}
A few things need to be addressed.
You need to use the ViewHolder pattern. That and the things you really need to know about ListView are in the GoogleIO 2010 - The World of ListView video.
It's not really a good idea to adjust the layout of an item in getView as that will negatively impact the view recycler that ListView uses. That is also in the video. I would really encourage you to try and find a different way to display what you want to display.
You're passing the x-dimension you want for an item in to your Adapter's constructor instead of supplying it an actual item resource id. ArrayAdapter will set your text for you if you pass in the right arguments. For a simple example using a string resource and the default provided list item layouts:
String[] mData = getResources().getStringArray(R.array.list_examples);
ArrayAdapter<String> mBaseAdapter =
new ArrayAdapter<String>(context,
android.R.layout.simple_list_item_1,
android.R.id.text1,
mListData);
If I were you, I'd just let the adapter set the text and grab that item layout by calling super like so:
@Override
public View getView(int position, View convertView, ViewGroup viewGroup){
convertView = super.getView(position,convertView, viewGroup);
//other code here
}
If you really want to do it yourself, here it is with the ViewHolder pattern
//This is the ViewHolder
static class Holder{
TextView mText;
}
@Override
public View getView(int position, View convertView, ViewGroup viewGroup){
convertView = super.getView(position, convertView, viewGroup);
if (convertView == null){
// If you reach this, it most likely means you're not supplying the
// adapter the layout and textview resource it needs. You can
// just comment out your getView override and see if your text gets
// displayed.
}
Holder mHolder;
if(convertView.getTag() == null){
mHolder = new Holder();
mHolder.mText = (TextView)convertView.findViewById(android.R.id.text1);
convertView.setTag(mHolder);
} else {
mHolder = (Holder) convertView.getTag();
}
mHolder.mText.setText(getItem(position).toString());
}
// These are both implemented in AdapterView which means
// they're flat list positions
int firstVis = mList.getFirstVisiblePosition();
int lastVis = mList.getLastVisiblePosition();
/* This is the "conversion" from flat list positions to ViewGroup child positions */
int count = lastVis - firstVis;
/* getChildAt(pos) is implemented in ViewGroup and has a different meaning for
* its position values. ViewGroup tracks visible items as children and is 0 indexed.
* This means you'll have 0 - X positions where X is however many items it takes
* to fill the visible area of your screen; usually less than 10. */
View listItem = mList.getChildAt(count - ((int)count/2)); // This will get the middle
// item for you to adjust as
// you want.
To find the item in the middle of the screen, you just need to take your visible item count and subtract half of that count as commented above.
You will want to set an AbsListView.OnScrollListener on your ListView and when the scroll stops, apply your adjustments to the middle item. Ex:
mList.setOnScrollListener(new OnScrollListener(){
@Override
public void onScrollStateChanged(AbsListView view, int scrollState){
}
@Override
public void onScroll(AbsListView view,
int firstVisibleItem,
int visibleItemCount,
int totalItemCount){
//Note that this is called when the scroll completes.
//It gives us the positions we want so just calculate
//the middle item, grab it and adjust the height directly
//or by increasing margins if the item layout_height is
//wrap_content. I'm not positive the margins do what you
// need, but setting the height directly should.
LinearLayout listItem = mList // <-- Replace LinearLayout with your layout
.getChildAt(visibleItemCount - ((int)visibleItemCount/2));
//Make sure you use the right LayoutParams for the listItem Layout
//If your listItem is a RelativeLayout for instance, you would use:
// listItem.setLayoutParams(
// new RelativeLayout.LayoutParams(LayoutParams.FILL_PARENT,
// theHeightYouWantHere));
listItem.setLayoutParams(
new LinearLayout.LayoutParams(LayoutParams.FILL_PARENT,
theHeightYouWantHere));
}
});
I've made a few tangentially related blog posts about ListView and ExpandableListView which have full code examples on how to refresh visible items (the example code just changes the text/background color of the item) which uses the childAt strategy. Pardon the personal blog link and the styling of it (it's new and needs more work :), but the code works well.
This does exactly what you want as long as you're using a TextView for the item layout (android.R.simple_list_item_1 is a TextView). If you decide to use a ViewGroup layout, you'll need to replace item.setHeight with the LayoutParams method.
mList.setOnScrollListener(new OnScrollListener(){
@Override
public void onScrollStateChanged(final AbsListView view, final int scrollState){
}
@Override
public void onScroll(final AbsListView view,
final int firstVisibleItem,
final int visibleItemCount,
final int totalItemCount){
if (visibleItemCount != 0){
Log.i( TAG, "firstVisibleItem: "
+ firstVisibleItem
+ "\nvisibleItemCount: "
+ visibleItemCount);
final int midPosition = visibleItemCount - (visibleItemCount / 2);
final TextView listItem = (TextView) mList.getChildAt(midPosition);
listItem.setHeight(500); //Only works if you're using
//android.R.simple_list_item_1 since it's
//not a ViewGroup, but rather a TextView
int count = visibleItemCount;
while (count >= 0){ // Here we need to loop through and make sure
// all recycled items are returned to their
// original height.
final TextView item = (TextView) mList.getChildAt(count);
if (item != null && count != midPosition){
item.setHeight(100);
}
count--;
}
}
}
});