1

我有一个与自定义 SimpleAdapter 关联的列表视图。我制作了自定义 SimpleAdapter,因为我想将不同的事件处理程序与包含的 textview 和 imageview 相关联,而不是其余的列表项小部件。所以,基本上我对列表项的两个部分有两个不同的事件处理程序。

我的 SimpleAdapter 自定义是:

class ClickableButtonListAdapter extends SimpleAdapter {

    private static class ViewHolder {
        TextView text;
        ImageView image;
    }

    public ClickableButtonListAdapter(Context context,
        List<? extends Map<String, ?>> data, int resource, String[] from, int[] to) {
        super(context, data, resource, from, to);
    }

    @SuppressWarnings("unchecked")
    @Override
    public View getView(final int position, View convertView, ViewGroup parent) {
        final View view = super.getView(position, convertView, parent);
        ViewHolder holder = (ViewHolder) view.getTag();
        if(holder == null) {
            holder = new ViewHolder();
            holder.text = (TextView) view.findViewById(R.id.comments);
            holder.image = (ImageView) view.findViewById(R.id.arrow);
            view.setTag(holder);
            final Context context = view.getContext();
            final HashMap<String, String> article = (HashMap<String,String>) getItem(position);
            OnClickListener listener = new OnClickListener() {
                @Override
                public void onClick(View view) {
                    String item_id = article.get("item_id");
                    Intent intent = new Intent(context, HNewsCommentsActivity.class);
                    intent.putExtra("item_id", item_id);
                    context.startActivity(intent);
                }
            };
            holder.text.setOnClickListener(listener);
            holder.image.setOnClickListener(listener);
        }
        return view;
    }
}

稍后,在我的 onCreate 活动中,我将我的自定义 SimpleAdapter 关联如下:

    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        try {
            final ArrayList<HashMap<String, String>> articles = getHNewsFeed();
            final SimpleAdapter adapter = new ClickableButtonListAdapter(this,
                    articles, R.layout.article,
                    new String[] {"title", "urlShort", "score", "comments", "item_id"},
                    new int[] {R.id.title, R.id.url, R.id.score, R.id.comments, R.id.item_id}
                    );
            final ListView l = (ListView) findViewById(android.R.id.list);
            l.setAdapter(adapter);
            l.setOnItemClickListener( new OnItemClickListener() {
                @Override
                public void onItemClick(AdapterView<?> parent, View v, int position, long id) {
                    final HashMap<String, String> article = articles.get(position);
                    String url = article.get("url");
                    Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(url));
                    startActivity(intent);
                }
            });
        } catch (Exception e) {
            Log.w(TAG, e.getMessage());
            Toast.makeText(this, e.getMessage(), Toast.LENGTH_SHORT).show();
        }
    }

我的问题是这些与包含的 textview 和 imageview 关联的事件处理程序(我在自定义 SimpleAdapter 的 getView 中初始化)似乎得到了错误的数据源项,并在单击时显示不同文章的数据。然而,listview 项单击处理程序获取正确的数据源项。有人可以帮我指出为什么我与自定义 SimpleAdapter 的 getView 中的 textview 和 imageview 关联的事件处理程序不能正常工作吗?我的理解是 SimpleAdapter 的 'getItem(position)' 方法应该返回正确的数据源项。但由于某种原因,它似乎没有这样做。

4

2 回答 2

4

你的逻辑有错误。Adapter的getView方法被调用很多次,一个View可以在不同的时间段显示不同的item。

那么它是如何工作的:

当 ListView 想要一个项目时,它会从 Adapter 类调用 getView 方法(无论该视图是否应该显示给用户或仅用于计算视图项目的大小)。ListView 传递 convertView 参数 - 这是此适配器前一段时间返回的旧视图,现在不需要了。可以重复使用它而不是再次膨胀(超级方法是这样的)。这是您的代码问题。如果将旧视图传递给您的 getView 方法,则它已经具有标记对象。而你的 (holder == null) == false。因此,代表一篇文章的视图打开了另一篇文章。

尝试改变你的逻辑来解决这个错误。最好的方法是只创建一次持有者,每次都设置视图监听器。

希望这会有所帮助。

编辑。这是解决问题的两种方法。

@SuppressWarnings("unchecked")
@Override
public View getView(final int position, View convertView, ViewGroup parent) {
    final View view = super.getView(position, convertView, parent);
    ViewHolder holder = (ViewHolder) view.getTag();
    if(holder == null) {
        holder = new ViewHolder();
        holder.text = (TextView) view.findViewById(R.id.comments);
        holder.image = (ImageView) view.findViewById(R.id.arrow);
        view.setTag(holder);
    }

    final Context context = view.getContext();
    final HashMap<String, String> article = (HashMap<String,String>) getItem(position);
    OnClickListener listener = new OnClickListener() {
        @Override
        public void onClick(View view) {
            String item_id = article.get("item_id");
            Intent intent = new Intent(context, HNewsCommentsActivity.class);
            intent.putExtra("item_id", item_id);
            context.startActivity(intent);
        }
    };
    holder.text.setOnClickListener(listener);
    holder.image.setOnClickListener(listener);

    return view;
}

第二,我更爱。感谢 Vinay S Shenoy 提供这个版本。

@SuppressWarnings("unchecked")
@Override
public View getView(final int position, View convertView, ViewGroup parent) {
    final HashMap<String, String> article = (HashMap<String,String>) getItem(position);

    final View view = super.getView(position, convertView, parent);
    ViewHolder holder = (ViewHolder) view.getTag();
    if(holder == null) {
        holder = new ViewHolder();
        holder.text = (TextView) view.findViewById(R.id.comments);
        holder.image = (ImageView) view.findViewById(R.id.arrow);
        view.setTag(holder);
        final Context context = view.getContext();
        OnClickListener listener = new OnClickListener() {
            @Override
            public void onClick(View view) {
                String item_id = view.getTag();
                //String item_id = article.get("item_id");
                Intent intent = new Intent(context, HNewsCommentsActivity.class);
                intent.putExtra("item_id", item_id);
                context.startActivity(intent);
            }
        };
        holder.text.setOnClickListener(listener);
        holder.image.setOnClickListener(listener);
    }

    holder.text.setTag(article.get("item_id"));
    holder.image.setTag(article.get("item_id"));

    return view;
}
于 2013-01-02T04:19:24.687 回答
2

问题是因为文章对象是在 OnClickListener 之外创建的。

您需要做的是将 item_id 作为标签保存在 TextView 和 ImageView 中。在 res/strings.xml 文件中添加两个字符串,其 ID 为 tag_holder 和 tag_item_id。然后,像这样更改您的 getView() 。

 @Override
    public View getView(final int position, View convertView, ViewGroup parent) {
        final View view = super.getView(position, convertView, parent);
        ViewHolder holder = (ViewHolder) view.getTag(R.string.tag_holder);
        if(holder == null) {
            holder = new ViewHolder();
            holder.text = (TextView) view.findViewById(R.id.comments);
            holder.image = (ImageView) view.findViewById(R.id.arrow);
            view.setTag(R.string.tag_holder, holder);
            final Context context = view.getContext();

            OnClickListener listener = new OnClickListener() {
                @Override
                public void onClick(View view) {
                    String item_id = (String)view.getTag(R.string.tag_item_id);
                    //String item_id = article.get("item_id");
                    Intent intent = new Intent(context, HNewsCommentsActivity.class);
                    intent.putExtra("item_id", item_id);
                    context.startActivity(intent);
                }
            };
            holder.text.setOnClickListener(listener);
            holder.image.setOnClickListener(listener);
        }

         final HashMap<String, String> article = (HashMap<String,String>) getItem(position);

            holder.text.setTag(R.string.tag_item_id, article.get("item_id");
            holder.image.setTag(R.string.tag_item_id, article.get("item_id");
        return view;
    }

此外,我建议不要为每个列表项创建新的 OnClickListener(),因为此 OnClick 侦听器仅依赖于 View Tag。相反,在适配器的构造函数中实例化您的侦听器,并对所有 List 项使用相同的实例。

于 2013-01-02T04:09:49.210 回答