2013-12-16 117 views
3

詢問問題並花費15天解決此問題後,我再次尋找幫助和解決方案。 在MainActivity中,我創建了Json Downloading Task,它從http下載數據,並使用CustomListAdapter.class填充listview。 一切正常。 現在,在列表視圖中,我有2個textview,我希望是可點擊的,其中一個是「接受」,textview只是在xml中沒有填充適配器或Json。 「接受」應該像這樣工作「將文本更改爲接受並更改顏色」及其工作方式與其他所有方式一樣。但當我點擊第一個「接受」(位置0)在列表視圖 它改變其他列表視圖項目(位置4,9)。這就像我點擊位置4,9的文字。 第一張圖片是在點擊「接受」之前,第二張圖片是在點擊之後。ListView子對象可點擊確認

before clicking /// after clicking

public class MainActivity extends Activity { 

protected static final String TAG = null; 
public ArrayList<FeedItem> feedList; 
public ListView feedListView; 
private ProgressBar progressbar; 
private CustomListAdapter adap; 
private LayoutInflater mInflater; 


@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
     super.onCreate(savedInstanceState); 
     setContentView(R.layout.activity_main); 
     feedListView= (ListView) findViewById(R.id.custom_list); 

     mInflater = (LayoutInflater) getSystemService(Activity.LAYOUT_INFLATER_SERVICE); 
     String url = "..."; 
     new DownloadFilesTask().execute(url); 

     getActionBar().setIcon(R.drawable.angel); 
     progressbar = (ProgressBar)findViewById(R.id.progressBar); 


     public void updateList() { 
    adap = new CustomListAdapter(this, feedList); 

      feedListView.setAdapter(adap); 

      } 


     public class DownloadFilesTask extends AsyncTask<String, Integer, Void> { 


     ///.... 

CustomListAdapter.class

public class CustomListAdapter extends BaseAdapter 
{ 

private ArrayList<FeedItem> listData; 
private LayoutInflater layoutInflater; 
private Context mContext; 
private ArrayList<String> data; 
protected ListView feedListView; 
ArrayList<HashMap<String,String>> list; 

public CustomListAdapter(Context context, ArrayList<FeedItem> listData) 
{ 
    this.listData = listData; 
    layoutInflater = (LayoutInflater) context 
      .getSystemService(Context.LAYOUT_INFLATER_SERVICE); 
    mContext = context; 
    data = new ArrayList<String>(); 
    for (int i = 0; i < 10; i++) { 
     data.add("Sample Text " + String.valueOf(i)); 
    } 
} 


@Override 
public int getCount() 
{ 
    return listData.size(); 
} 

@Override 
public Object getItem(int position) 
{ 
    return listData.get(position); 
} 

@Override 
public long getItemId(int position) 
{ 
    return position; 
} 


public View getView(int position, View convertView, ViewGroup parent) 
{ 
final ViewHolder holder; 
View row=convertView; 
    if ((row == null) || (row.getTag()==null)) { 

    convertView = layoutInflater.inflate(R.layout.list_row_layout, null); 
    holder = new ViewHolder(); 
    holder.headlineView = (TextView)convertView.findViewById(R.id.name); 
    holder.reportedDateView = (TextView) convertView.findViewById(R.id.confid); 
    holder.accept= (TextView) convertView.findViewById(R.id.acceptTV); 

    convertView.setTag(holder); 



    } 
    else 
    { 
     holder = (ViewHolder) convertView.getTag(); 

    } 

    final FeedItem newsItem = (FeedItem) listData.get(position); 
    holder.accept.setFocusable(true); 

    holder.accept.setClickable(true); 
    holder.headlineView.setText(Html.fromHtml(newsItem.getTitle())); 
    holder.reportedDateView.setText(Html.fromHtml(newsItem.getContent())); 

    holder.accept.setOnClickListener(new View.OnClickListener() { 

     @Override 
     public void onClick(View arg0) { 


        holder.accept.setText(Html.fromHtml(newsItem.getContent())); 
     } 
    }); 



    return convertView; 
} 




static class ViewHolder 
{ 

    TextView accept; 
    TextView headlineView; 
    TextView reportedDateView; 
    ImageView imageView; 
    FeedItem newsItem; 

} 
+0

卸下支架,然後用 – Rohit

+0

只爲「接受」的TextView或所有? –

+0

@Rohit爲什麼ViewHolder有問題? – Raghunandan

回答

6

您需要了解的循環機制是如何運作的ListView

How ListView's recycling mechanism works

使用一個模型類。假設你已經有了下面的

public class FeedItem { 

String title,content; 

public String getTitle() { 
    return title; 
} 

public void setTitle(String title) { 
    this.title = title; 
} 

public String getContent() { 
    return content; 
} 

public void setContent(String content) { 
    this.content = content; 
} 

} 

getView

holder.accept.setText(listData.get(position).getContent()); 
holder.accept.setTag(position); 
holder.accept.setOnClickListener(mClickListener); 

然後

private OnClickListener mClickListener = new OnClickListener() { 
public void onClick(View v) { 
    int pos = (Integer) v.getTag(); 
    FeedItem newsItem = (FeedItem) listData.get(pos); 
    newsItem.setContent("Accepted"); 
    CustomListadapter.this.notifyDataSetChanged(); 
} 
}; 

Exaplanation:

您使用具有getter和setter模型類。

setTag到位置的按鈕。在onClick你得到標籤即位置並相應地改變內容。通過在適配器上調用notifyDataSetChanged來刷新列表視圖。

對於其他人在這裏的好處就是一個例子

public class MainActivity extends Activity { 

    ArrayList<Holder> list = new ArrayList<Holder>(); 
    ListView lv; 
    CustomListAdapter cus; 
    @Override 
    protected void onCreate(Bundle savedInstanceState) { 
     super.onCreate(savedInstanceState); 
     setContentView(R.layout.activity_main); 
     lv = (ListView) findViewById(R.id.listView1); 
     for(int i=0;i<10;i++) 
     { 
      Holder h = new Holder(); 
      h.setTitle("Title"+i); 
      h.setContent("Content"+i); 
      h.setColor(Color.BLACK); 
      list.add(h); 
     } 
     cus = new CustomListAdapter(this,list); 
     lv.setAdapter(cus); 
    } 
} 

Model類持有人

public class Holder { 

    String title,content; 
    int color; 

    public int getColor() { 
return color; 

    public void setColor(int color) { 
this.color = color; 
    } 

    public String getTitle() { 
     return title; 
    } 

    public void setTitle(String title) { 
     this.title = title; 
    } 

    public String getContent() { 
     return content; 
    } 

    public void setContent(String content) { 
     this.content = content; 
    } 

} 

CustomListAdapter

public class CustomListAdapter extends BaseAdapter{ 

    LayoutInflater inflater; 
    ArrayList<Holder> list; 
    public CustomListAdapter(MainActivity mainActivity, ArrayList<Holder> list) { 
     inflater = LayoutInflater.from(mainActivity); 
     this.list =list; 
    } 
    @Override 
    public int getCount() { 
     // TODO Auto-generated method stub 
     return list.size(); 
    } 
    @Override 
    public Object getItem(int position) { 
     // TODO Auto-generated method stub 
     return position; 
    } 
    @Override 
    public long getItemId(int position) { 
     // TODO Auto-generated method stub 
     return position; 
    } 
    public View getView(int position, View convertView, ViewGroup parent) { 
     ViewHolder holder; 
     if (convertView == null) { 
      convertView = inflater.inflate(R.layout.list_item, 
        parent, false); 
      holder = new ViewHolder(); 
      holder.tv = (TextView) convertView.findViewById(R.id.textView1); 
      holder.b = (Button) convertView.findViewById(R.id.button1); 
      convertView.setTag(holder); 
     } else { 
      holder = (ViewHolder) convertView.getTag(); 
     } 
     Holder h = list.get(position); 
     holder.tv.setText(h.getTitle()); 
     holder.b.setText(h.getContent()); 
     holder.b.setTextColor(h.getColor()); 
     holder.b.setOnClickListener(mClickListener); 
     holder.b.setTag(position); 
     return convertView; 
} 
    private OnClickListener mClickListener = new OnClickListener() { 

      public void onClick(View v) { 
       int pos = (Integer) v.getTag(); 
       Holder h = (Holder) list.get(pos); 
       h.setContent("Accepted"); 
        h.setColor(Color.BLUE); 
       CustomListAdapter.this.notifyDataSetChanged(); 

      } 

      }; 
    static class ViewHolder 
    { 
     TextView tv; 
     Button b; 
    } 
} 

list_item.xml

​​3210點擊

activity_main.xml中

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" 
    xmlns:tools="http://schemas.android.com/tools" 
    android:layout_width="match_parent" 
    android:layout_height="match_parent" 
    android:paddingBottom="@dimen/activity_vertical_margin" 
    android:paddingLeft="@dimen/activity_horizontal_margin" 
    android:paddingRight="@dimen/activity_horizontal_margin" 
    android:paddingTop="@dimen/activity_vertical_margin" 
    tools:context=".MainActivity" > 
    <ListView 
     android:id="@+id/listView1" 
     android:layout_width="match_parent" 
     android:layout_height="wrap_content" 
     > 
    </ListView> 

</RelativeLayout> 

在行1和5按鍵使其改變爲接受和是藍色的。

enter image description here

+0

THAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAANK YOU !!!!!!!!!!!!!!!!!!!!!!!!! 15天后......你是天使! –

+0

謝謝@Raghunandan,我也在我的應用程序中進行了更改,並從我+1 – Rohit

0

你的問題的事實,你不復位的意見狀態,他們被重用,當談到肯定。 ListView重用視圖以獲得更好的內存管理。因此,請根據物品的選定狀態調整您的getView()以調整文字和顏色。我沒有看到你的代碼保存在哪裏,所以我不能給你一個工作的例子。

喜歡的東西

if (newsItem.selected) 
     holder.accept.setText("accepted"); 
    else 
     holder.accept.setText("accept"); 
1
getView(...){ 
if ((row == null) || (row.getTag()==null)) { 
// some code 
}else{ 
// some code 
} 
holder.accept.setTag(position); 

// some more code 

if(newsItem.isSelected()){ 
holder.accept.setText("accepted"); 
}else{ 
holder.accept.setText("accept"); 
} 

//handling click 
holder.accept.setOnClickListener(new View.OnClickListener() { 

    @Override 
    public void onClick(View arg0) { 

       int position = (Integer)arg0.getTag(); 
       // change backing dataset here instead. 
       FeedItem m = listData.get(position); 
       // declare a boolean 'selected' in FeedItem 
       // toggle the previous selection 
       m.setSelected(! m.isSelected()); 
       // call notifydatasetChanged 
       CustomListAdapter.this.notifyDataSetChanged(); 

    } 
}); 

// some more code 
} 


class FeedItem{ 
// some data member 

boolean selected = false; 
public boolean isSelected(){ 
return selected; 
} 

public void setSelected(boolean status){ 
selected = status; 
} 
} 
+0

這是正確的。 FeetItem是模型類。但你必須在這裏更改數據 – Raghunandan

+0

@Raghunandan,我正在改變選擇狀態並且設置爲true。或者,我認爲切換狀態會更好。 – guptakvgaurav

+0

op是設置文本到按鈕,而不是切換更改按鈕settext。任何方式,這將改變文字也 – Raghunandan