Android ListFragment дублирует записи
У меня есть это приложение, где я реализую интерфейс фрагмента ActionBar. В основе интерфейса лежит один ListFragment, где записи в списке дублируются каждый раз, когда я переключаюсь на другую вкладку. Может кто-нибудь показать мне правильный способ реализации фрагмента списка с помощью специального адаптера?
Вот мой код: FeaturedFragment.java
public class FeaturedFragment extends ListFragment {
static final String URL = "http://www.sundancepost.com/ivue/Featured.xml";
static final String KEY_PROJECT = "project"; // parent node
static final String KEY_BANNER = "banner";
ListView featuredList;
FeaturedListAdapter featuredAdapter;
ArrayList<HashMap<String, String>> projectsList = new ArrayList<HashMap<String, String>>();
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
super.onCreateView(inflater, container, savedInstanceState);
View view = inflater.inflate(R.layout.featured_list, container, false);
featuredList = (ListView)view.findViewById(android.R.id.list);
return view;
}
@Override
public void onActivityCreated(Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
ConnectivityManager cMgr = (ConnectivityManager)getActivity().getSystemService(Context.CONNECTIVITY_SERVICE);
if (cMgr.getActiveNetworkInfo() != null && cMgr.getActiveNetworkInfo().isConnectedOrConnecting()) {
new MyAsyncTask().execute();
} else {
AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());
builder.setMessage("Please check your internet connection");
builder.setTitle("Failed to download resources");
builder.setCancelable(false);
builder.setPositiveButton("Ok", new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int id) {
return;
}
});
AlertDialog alert = builder.create();
alert.show();
}
featuredAdapter = new FeaturedListAdapter(getActivity(), projectsList);
}
public class MyAsyncTask extends AsyncTask<Void,Void,Void>{
private final ProgressDialog recents_dialog = new ProgressDialog(getActivity());
@Override
protected Void doInBackground(Void... params) {
XMLParser parser = new XMLParser();
String xml = parser.getXmlFromUrl(URL);
Document doc = parser.getDomElement(xml);
NodeList nl = doc.getElementsByTagName(KEY_PROJECT);
for (int i = 0; i < nl.getLength(); i++) {
HashMap<String, String> map = new HashMap<String, String>();
Element e = (Element) nl.item(i);
map.put(KEY_BANNER, parser.getValue(e, KEY_BANNER));
projectsList.add(map);
}
return null;
}
@Override
protected void onPreExecute()
{
recents_dialog.setMessage("Loading ...");
recents_dialog.show();
recents_dialog.setCancelable(false);
}
@Override
protected void onPostExecute(Void result)
{
if(recents_dialog.isShowing() == true)
{
recents_dialog.dismiss();
}
// Getting adapter by passing xml data ArrayList
featuredList.setAdapter(featuredAdapter);
}
}
}
FeaturedListAdapter:
public class FeaturedListAdapter extends BaseAdapter {
private Activity activity;
ArrayList<HashMap<String, String>> data;
private static LayoutInflater inflater=null;
public ImageLoader imageLoader;
public FeaturedListAdapter(Activity a, ArrayList<HashMap<String, String>> d) {
activity = a;
data= (ArrayList<HashMap<String, String>>) d;
inflater = (LayoutInflater)activity.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
imageLoader=new ImageLoader(activity.getApplicationContext());
}
public int getCount() {
return data.size();
}
public Object getItem(int position) {
return position;
}
public long getItemId(int position) {
return position;
}
public View getView(int position, View convertView, ViewGroup parent) {
View vi = convertView;
if(vi==null){
vi=new View(activity);
vi = inflater.inflate(R.layout.featured_listrow, null);
}
ImageView banner =(ImageView)vi.findViewById(R.id.banner); // thumb image
banner.setAdjustViewBounds(true);
imageLoader.DisplayImage(data.get(position).get(FeaturedFragment.KEY_BANNER), banner);
return vi;
}
}
[Редактировать] Следующий код изменен в соответствии с предложением @10s. Но результат все тот же..
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
if (convertView == null) {
convertView = inflater.inflate(R.layout.featured_listrow, null);
holder = new ViewHolder();
holder.banner = (ImageView)convertView.findViewById(R.id.banner);
holder.banner.setAdjustViewBounds(true);
imageLoader.DisplayImage(data.get(position).get(FeaturedFragment.KEY_BANNER), holder.banner);
convertView.setTag(holder);
} else {
holder = (ViewHolder)convertView.getTag();
}
return convertView;
}
public static class ViewHolder {
public ImageView banner;
}
2 ответа
Код, который вы используете в getView() пользовательского адаптера, воссоздает представление при каждом его вызове. А адаптер перерисовывает себя каждый раз, когда ваша активность, в вашем случае фрагмент фокусируется.
Попробуйте использовать держатель для вашего адаптера. Это лучшая практика для Android и очень простая и полезная. Ссылка для начала работы: http://android.amberfog.com/?p=296
Причина добавления повторяющихся записей при каждом переключении вкладки заключается в том, что метод onActivityCreated() вызывается каждый n при каждом переключении на эту вкладку. И, следовательно, ваш AsyncTask выполняется каждый раз, и в методе doInBackground() AsyncTask вы продолжали добавлять свой элемент в arraylist projectList, не очищая его.
Очистите projectList перед запуском asyncTask или проверьте, содержат ли список некоторые данные или нет, и на основании этого сделайте вызов для выполнения asyncTask (если list равен нулю или не содержит каких-либо элементов).