从数据库以listview的形式显示的数据(此处为listview,其中的标头已禁用标头的onClick)。
我试图从显示所选项目的描述position
的getView()
。该列表非常大,因此它在滚动时会动态分配视图,并在滚动position
后给出错误的值
我观看了Google I / O 2010-ListView视频的世界,它阐明了这些内容。
因此,我认为我需要实现notifyDataSetChanged()
或者onScroll()
,onScrollStateChanged()
方法。
但是如何?
代码:
public class MainActivity1 extends ListActivity implements OnTouchListener{
private MyCustomAdapter mAdapter;
Activity temp = this;
String []s = new String[500];
ArrayList<GS> q = new ArrayList<GS>();
CustomAdapter adapter;
ListView lv;
int c=1;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
DBAdapter db = DBAdapter.getDBAdapter(getApplicationContext());
if (!db.checkDatabase())
{
db.createDatabase(getApplicationContext());
}
db.openDatabase();
q = db.getData();
mAdapter = new MyCustomAdapter();
mAdapter.addSeparatorItem(q.get(0).getA_name());
mAdapter.addItem(q.get(0).getAS_name());
for (int i = 1; i < 460; i++) {
if (!(q.get(i).getA_name().trim().equals(q.get(i-1).getA_name().trim()))) {
mAdapter.addSeparatorItem(q.get(i).getA_name());
c++;
}
mAdapter.addItem(q.get(i).getAS_name());
}
setListAdapter(mAdapter);
}
//Adapter Class
private class MyCustomAdapter extends BaseAdapter {
private static final int TYPE_ITEM = 0;
private static final int TYPE_SEPARATOR = 1;
private static final int TYPE_MAX_COUNT = TYPE_SEPARATOR + 1;
private ArrayList<String> mData = new ArrayList<String>();
private LayoutInflater mInflater;
private TreeSet<Integer> mSeparatorsSet = new TreeSet<Integer>();
public MyCustomAdapter() {
mInflater = (LayoutInflater)getSystemService(Context.LAYOUT_INFLATER_SERVICE);
}
public void addItem(final String item) {
mData.add(item);
notifyDataSetChanged();
}
public void addSeparatorItem(final String item) {
mData.add(item);
// save separator position
mSeparatorsSet.add(mData.size() - 1);
notifyDataSetChanged();
}
@Override
public int getItemViewType(int position) {
return mSeparatorsSet.contains(position) ? TYPE_SEPARATOR : TYPE_ITEM;
}
@Override
public int getViewTypeCount() {
return TYPE_MAX_COUNT;
}
public int getCount() {
return mData.size();
}
public String getItem(int position) {
return mData.get(position);
}
public long getItemId(int position) {
Log.v("getItemId Position", ""+position);
return position;
}
public View getView(final int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
int type = getItemViewType(position);
// System.out.println("getView " + position + " " + convertView + " type = " + type);
if (convertView == null) {
holder = new ViewHolder();
switch (type) {
case TYPE_ITEM:
convertView = mInflater.inflate(R.layout.activity_main1, null);
holder.textView = (TextView)convertView.findViewById(R.id.text);
break;
case TYPE_SEPARATOR:
convertView = mInflater.inflate(R.layout.activity_main2, null);
holder.textView = (TextView)convertView.findViewById(R.id.textSeparator);
count++;
break;
}
convertView.setTag(holder);
} else {
holder = (ViewHolder)convertView.getTag();
}
holder.textView.setText(mData.get(position));
// We set the OnClickListener here because it is unique to every
// item. Although views can be recycled & reused, an OnClickListener cannot be.
if (type == TYPE_ITEM) {
holder.textView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
AlertDialog.Builder x = new AlertDialog.Builder(
temp);
Log.v("position",""+position);
x.setIcon(R.drawable.ic_launcher)
.setTitle(q.get(position-count).getAS_name())
.setMessage(q.get(position-count).getDesc_art())
.setCancelable(true)
.setPositiveButton("OK",
new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface arg,
int arg1) {
}
});
AlertDialog a = x.create();
a.show();
}
});
} else {
holder.textView.setOnClickListener(null);
count++;
}
return convertView;
}
}
public static class ViewHolder {
public TextView textView;
}
public boolean onTouch(View v, MotionEvent event) {
// TODO Auto-generated method stub
return false;
}
}
所以,我试图单击时在Alertdialog中显示所选项目的TYPE_ITEM
描述。每个TYPE_ITEM的描述顺序存储在数据库中。因此,我需要跟踪TYPE_ITEM的位置(意味着建立索引),以便从数据库中获取数据。我希望你现在了解我的问题!
顺便说一句,我试图在getView()
其中使用一个count变量,但是如果我们再次向上滚动(在这种情况下,count
应该减少),该怎么办。我希望您现在了解问题&我认为我们应该实现notifyDataSetChanged或onScroll,onScrollStateChanged方法。
如果问题不清楚,那么您可以在评论中问我
请帮助!
您需要做的就是将OnClickListener
on的设置移动holder.textview
。
这里的问题是:
:由于您在if (convertView == null)
块中设置了OnClickListener ,因此当convertView不为null时,它不会更改。因此,position
inAlertDialog
中使用的是您convertView
为null时设置的。
:解决方案是在每次处理职位时都设置OnClickListener-我们不在乎视图是否被回收!!!我们需要重置OnClickListener以反映正确/当前位置。
:尽管我们从来没有在item为时在holder.textview上设置OnClickListener,但是使用可以TYPE_SEPARATOR
安全地删除OnClickListener holder.textview.setOnClickListener(null)
。
请尝试下面的更新代码。我希望这里的逻辑很清楚。
//Adapter Class
private class MyCustomAdapter extends BaseAdapter {
....
....
public View getView(final int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
int type = getItemViewType(position);
System.out.println("getView " + position + " " + convertView + " type = " + type);
if (convertView == null) {
holder = new ViewHolder();
switch (type) {
case TYPE_ITEM:
convertView = mInflater.inflate(R.layout.activity_main1, null);
holder.textView = (TextView)convertView.findViewById(R.id.text);
break;
case TYPE_SEPARATOR:
convertView = mInflater.inflate(R.layout.activity_main2, null);
holder.textView = (TextView)convertView.findViewById(R.id.textSeparator);
break;
}
convertView.setTag(holder);
} else {
holder = (ViewHolder)convertView.getTag();
}
holder.textView.setText(mData.get(position));
// We set the OnClickListener here because it is unique to every
// item. Although views can be recycled & reused, an OnClickListener cannot be.
if (type == TYPE_ITEM) {
holder.textView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
AlertDialog.Builder x = new AlertDialog.Builder(
temp);
Log.v("position",""+position);
x.setIcon(R.drawable.ic_launcher)
.setTitle(q.get(position-1).getAS_name())
.setMessage(q.get(position-1).getDesc_art())
.setCancelable(true)
.setPositiveButton("OK",
new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface arg,
int arg1) {
}
});
AlertDialog a = x.create();
a.show();
}
});
} else {
holder.textview.setOnClickListener(null);
}
return convertView;
}
....
....
}
编辑:
包装器类(可以实现为的内部类,MainActivity1
也可以独立地实现):
public class ContentWrapper {
private String mItem, mItemDescription;
public ContentWrapper(String item, String itemDescription) {
mItem = item;
mItemDescription = itemDescription;
}
public String getItem() {
return mItem;
}
public String getItemDescription() {
return mItemDescription;
}
}
您的数据设置也会更改:
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
DBAdapter db = DBAdapter.getDBAdapter(getApplicationContext());
if (!db.checkDatabase())
{
db.createDatabase(getApplicationContext());
}
db.openDatabase();
q = db.getData();
mAdapter = new MyCustomAdapter();
// mAdapter.addSeparatorItem(q.get(0).getA_name());
// First separator item
// No description
mAdapter.addSeparatorItem(new ContentWrapper(q.get(0).getA_name(), null));
// mAdapter.addItem(q.get(0).getAS_name());
// First TYPE_ITEM
// Pass the description
mAdapter.addItem(new ContentWrapper(q.get(0).getAS_name(), q.get(0).getDesc_art()));
for (int i = 1; i < 460; i++) {
if (!(q.get(i).getA_name().trim().equals(q.get(i-1).getA_name().trim()))) {
// mAdapter.addSeparatorItem(q.get(i).getA_name());
mAdapter.addSeparatorItem(new ContentWrapper(q.get(i).getA_name(), null));
c++;
}
// mAdapter.addItem(q.get(i).getAS_name());
mAdapter.addItem(new ContentWrapper(q.get(i).getAS_name(), q.get(i).getDesc_art()));
}
setListAdapter(mAdapter);
}
接下来,我们对适配器进行更改:
// private ArrayList<String> mData = new ArrayList<String>();
private ArrayList<ContentWrapper> mData = new ArrayList<ContentWrapper>();
该add*
方法
public void addItem(ContentWrapper value) {
mData.add(value);
notifyDataSetChanged();
}
public void addSeparatorItem(ContentWrapper value) {
mData.add(value);
// save separator position
mSeparatorsSet.add(mData.size() - 1);
notifyDataSetChanged();
}
public ContentWrapper getItem(int position) {
return mData.get(position);
}
该getView(...)
方法:
@Override
public View getView(final int position, View convertView, ViewGroup parent) {
....
....
holder.textView.setText(mData.get(position).getItem());
if (type == TYPE_ITEM) {
holder.textView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
AlertDialog.Builder x = new AlertDialog.Builder(temp);
Log.v("position",""+position);
x.setIcon(R.drawable.ic_launcher)
// .setTitle(q.get(position-count).getAS_name())
.setTitle(mData.get(position).getItem())
// .setMessage(q.get(position-count).getDesc_art())
.setMessage(mData.get(position).getItemDescription())
.setCancelable(true)
.setPositiveButton("OK",
new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface arg,
int arg1) {
}
});
AlertDialog a = x.create();
a.show();
}
});
} else {
holder.textView.setOnClickListener(null);
}
}
就是这样。
[我]认为我们应该实现notifyDataSetChanged或onScroll,onScrollStateChanged方法。
notifyDataSetChanged()
用于告诉适配器基础数据已更改,因此需要刷新。例如,如果某项的描述发生更改,则可以在中更新该项mData
并调用notifyDataSetChanged()
。但是根据您的情况(以及代码告诉我的情况),在使用-设置适配器之前,将初始化数据setListAdapter(mAdapter)
。因此,甚至不需要调用方法notifyDataSetChanged()
内部add*
。notifyDataSetChanged()
在将适配器附加到列表视图之前进行调用不会执行任何操作。
onScroll
并且onScrollChanged
是为了不同的目的。例如,假设您Go To Top of the List
在用户滚动到第50个项目时显示一个按钮-在用户向上滚动第50个位置时将其隐藏。在您的情况下,问题在于您试图从两个不同的源(mData,q)获取数据,并且同步出现问题。没有其他的。
本文收集自互联网,转载请注明来源。
如有侵权,请联系[email protected] 删除。
我来说两句