I have a gridview which shows all apps installed on the device.
Whenever I try to recycle items (Using convertView == null etc.) the same icon and Text shows up multiple times in my grid.
This is my code:
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ImageView imageView;
TextView txtName;
FrameLayout layout;
// if (convertView == null) {
imageView = new ImageView(myContext);
imageView.setLayoutParams(new GridView.LayoutParams(230, 230));
imageView.setScaleType(ImageView.ScaleType.CENTER_CROP);
imageView.setPadding(4, 4, 4, 4);
txtName = new TextView(myContext);
RelativeLayout.LayoutParams relativeParams = new RelativeLayout.LayoutParams(LayoutParams.FILL_PARENT,
LayoutParams.FILL_PARENT);
relativeParams.addRule(RelativeLayout.ALIGN_PARENT_BOTTOM);
txtName.setLayoutParams(relativeParams);
txtName.setGravity(Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL);
// txtName.setPadding(8, 8, 8, 8);
layout = new FrameLayout(myContext);
// FrameLayout.LayoutParams layoutparams = new
// FrameLayout.LayoutParams(ViewGroup.LayoutParams.FILL_PARENT,
// ViewGroup.LayoutParams.FILL_PARENT, Gravity.CENTER_HORIZONTAL |
// Gravity.CENTER_VERTICAL);
// layout.setLayoutParams(layoutparams);
RelativeLayout layout2 = new RelativeLayout(myContext);
layout.addView(imageView);
layout.addView(layout2);
layout2.addView(txtName);
ResolveInfo resolveInfo = MyAppList.get(position);
imageView.setImageDrawable(resolveInfo.loadIcon(myPackageManager));
txtName.setText(resolveInfo.loadLabel(myPackageManager));
// } else {
// layout = (FrameLayout) convertView;
// }
return layout;
}
If I don't recycle them, their shown correctly but the gridview is very slow.
And my second question is: How can I put the text UNDER the icon inside my gridview? Like in the native Android launcher?
This is my GridActivity:
public class AppListActivity extends Activity {
PackageManager myPackageManager;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
myPackageManager = getPackageManager();
Intent intent = new Intent(Intent.ACTION_MAIN, null);
intent.addCategory(Intent.CATEGORY_LAUNCHER);
List<ResolveInfo> appIntentList = getPackageManager()
.queryIntentActivities(intent, 0);
GridView gridview = new GridView(this);
gridview.setStretchMode(GridView.STRETCH_COLUMN_WIDTH);
gridview.setNumColumns(4);
gridview.setHorizontalSpacing(30);
gridview.setVerticalSpacing(30);
gridview.setBackgroundColor(getResources().getColor(android.R.color.black));
gridview.setGravity(Gravity.CENTER);
gridview.setColumnWidth(60);
gridview.setPadding(10, 10, 10, 10);
gridview.setAdapter(new MyAdapter(this, appIntentList));
gridview.setOnItemClickListener(new OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view,
int position, long id) {
ResolveInfo cleckedResolveInfo = (ResolveInfo) parent
.getItemAtPosition(position);
ActivityInfo clickedActivityInfo = cleckedResolveInfo.activityInfo;
Intent intent = new Intent(Intent.ACTION_MAIN);
intent.addCategory(Intent.CATEGORY_LAUNCHER);
intent.setClassName(
clickedActivityInfo.applicationInfo.packageName,
clickedActivityInfo.name);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK
| Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED);
startActivity(intent);
}
});
setContentView(gridview);
}
}
Well, you need to move the data fill of your list item outside of list item setup. Something like:
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ImageView imageView;
TextView txtName;
RelativeLayout layout;
if (convertView == null) {
imageView = new ImageView(myContext);
imageView.setLayoutParams(new GridView.LayoutParams(230, 230));
imageView.setScaleType(ImageView.ScaleType.CENTER_CROP);
imageView.setPadding(4, 4, 4, 4);
generateAndSetViewId(imageView);
txtName = new TextView(myContext);
RelativeLayout.LayoutParams relativeParams = new RelativeLayout.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
relativeParams.addRule(RelativeLayout.ALIGN_PARENT_BOTTOM);
relativeParams.addRule(RelativeLayout.BELOW, imageView.getId());
txtName.setLayoutParams(relativeParams);
txtName.setGravity(Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL);
// txtName.setPadding(8, 8, 8, 8);
layout = new RelativeLayout(myContext);
// FrameLayout.LayoutParams layoutparams = new
// FrameLayout.LayoutParams(ViewGroup.LayoutParams.FILL_PARENT,
// ViewGroup.LayoutParams.FILL_PARENT, Gravity.CENTER_HORIZONTAL |
// Gravity.CENTER_VERTICAL);
// layout.setLayoutParams(layoutparams);
layout.addView(imageView);
layout.addView(txtName);
} else {
layout = (RelativeLayout) convertView;
imageView = (ImageView) layout.getChildAt(0);
txtName = (TextView) layout.getChildAt(1);
}
ResolveInfo resolveInfo = MyAppList.get(position);
imageView.setImageDrawable(resolveInfo.loadIcon(myPackageManager));
txtName.setText(resolveInfo.loadLabel(myPackageManager));
return layout;
}
private static final AtomicInteger sNextGeneratedId = new AtomicInteger(1);
@SuppressLint("NewApi")
public static int generateAndSetViewId(View view) {
int viewID = -1;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
viewID = View.generateViewId();
} else {
while (true) {
final int result = sNextGeneratedId.get();
// aapt-generated IDs have the high byte nonzero; clamp to the range under that.
int newValue = result + 1;
if (newValue > 0x00FFFFFF) {
newValue = 1; // Roll over to 1, not 0.
}
if (sNextGeneratedId.compareAndSet(result, newValue)) {
viewID = result;
break;
}
}
}
view.setId(viewID);
return viewID;
}
Above code can be simplified a lot if you inflate the views from an XML instead of creating them dynamically.