Android 将自定义对象与 ArrayAdapter 结合使用,并传达返回视图

Android using custom object with ArrayAdapter, and communicating back views

有一些与此类似的问题,但我找不到我想要的问题。我正在使用存储在 ArrayAdapter 中的自定义 class 来填充具有线性布局的 ListView,我可以通过修改 class 进行自定义 - 这是因为我在每个按钮中都有一个按钮,它完全符合我的要求想。 LinearLayout 看起来像:[TextView][EditText][Button] 当单击 Button 时,它会使用 EditText 的内容执行一些数据库操作。没关系。

我遇到的问题是,我想(例如)在我的 activity 而不是 class 中更新 ListView 中 1 布局中 EditText 的文本内容创建视图并设置 Button 侦听器等。我可能还想说“禁用列表视图第二项中的按钮”等等。

我尝试了很多不同的方法,在适配器上传递上下文或调用方法与在实际对象列表上传递方法,使 EditText 成为具有 getter 的成员变量,我总是得到要么没有任何反应,要么在按 ID 查找 EditText 时出现 NullPointerException。

这可能吗?我是否需要创建一个最终变量来保存每个 EditText 并以某种方式将其传递回 activity 或其他东西?我将在我的 activity 中得到多个具有相同 ID 的视图,但它们将由不同的对象创建

最终我想做的是制作它,这样我就可以稍微抽象一下我的 GUI。目前我有 8 个活动,大约 90% 相同,但我有重复的代码在每个 activity 中设置视图。我想将其抽象为 classes,我可以说“给我一个带有任何标签的新文本输入行,我将覆盖按钮行为”,然后为此重构为 1 activity任务。当用户在 'rows' 之一中输入数据时, activity 可能需要更新其他行之一(或在按下后退按钮时清除 1 等),这就是我想要的原因能够回调到数组适配器内对象布局内的特定视图:D

例如,我希望能够创建 3 个新的 textInputLineBuilder 对象(最终可能是 subclass)并将它们显示为:

[名称:][NameEditText][按钮]

[Stock:][StockEditText][按钮]

[数量:][QuantityEditText][按钮]

这是(缩减的)布局,activity_row.xml 自定义对象膨胀,returns 到 ArrayAdapter

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout>
    <TextView
        android:id="@+id/rowLabel"/>    
    <EditText
        android:id="@+id/rowInput"/>    
    <Button
        android:id="@+id/rowButton"/>
</LinearLayout>

这里是activity中的相关代码:

public class AnActivity implements UICallBack {
    private ListView listOfInputs;
    private ArrayList<UIElementInterface> listOfUIElements;
    private UIElementsAdapter UIAdapter;
    
    protected void onCreate() {
        listOfInputs = (ListView) findViewById(R.id.listInputs);
        
        listOfUIElements = new ArrayList<>();
        listOfUIElements.add(new textInputLineBuilder("<label>", this));    //this implements a callback interface
        //add a few more

        UIAdapter = new UIElementsAdapter(this.getApplicationContext(), listOfUIElements);
        UIAdapter.notifyDataSetChanged();
        listOfInputs.setAdapter(UIAdapter);
    }
    
    public void updateUIState() {
        //I have tried all sorts here
        for (int i=0; i<listOfUIElements.size(); i++) {
            UIAdapter.setTheText("test "+i, i);
        }

        UIAdapter.notifyDataSetChanged;
        listOfInputs.setAdapter(UIAdapter);
    }
}

然后我有我的 ArrayAdapter class,UIElementsAdapter.java 从列表中检索适当的对象,然后调用 getRowView 以使对象膨胀将创建的布局这个 'row'

public class UIElementsAdapter extends ArrayAdapter<UIElementInterface> {
    private final Context context;

    public UIElementsAdapter(Context context, ArrayList<UIElementInterface> listOfUIElements) {
        super(context, 0, listOfUIElements);
        this.context = context;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        return getItem(position).getRowView(context, parent);   //retrieves the UIElementInterface object aka textInputLineBuilder object
    }

    public void setTheText(String s, Integer position) {
        try {
            getItem(position).setEditText(s);
        } catch(NullPointerException e) {
            e.printStackTrace();
        }
    }
}

最后我 textInputLineBuilder.java 实现了 UIElementInterface(getRowView 和 setEditText)

public class textInputLineBuilder implements UIElementInterface {
    private View convertView;

    public textInputLineBuilder(String label, UICallBack callback) {
        this.labelText = label;
        this.callback = callback;   //used with button onclicklistener
    }
    
    public View getRowView(Context context, ViewGroup parent) {     
        LayoutInflater layout = LayoutInflater.from(context);
        convertView = layout.inflate(R.layout.activity_row, parent, false);

        label = (TextView) convertView.findViewById(R.id.rowLabel);
        fuzzy = (Button) convertView.findViewById(R.id.rowButton);
        input = (EditText) convertView.findViewById(R.id.rowInput);
        final EditText inputFinal = input;    //for the button listener.  was trying using input as a member variable for setEditText but that didn't work
        
        label.setText(labelText);
        
        //add onclicklistener for the fuzzy button. works fine
        
        return convertView;
    }
    
    public void setEditText(String s) {
        //I have tried all sorts here, passing in context from various places, passing in layout inflaters etc
        EditText e = (EditText) convertView.findViewById(R.id.rowInput);
        e.setText(s);
    }
}

updateUIState() 中,您当前尝试通过调用 setTheText() 直接修改 EditText。使用调试器,我能够看到这在一定程度上起作用——EditTextsetEditText() 的末尾具有所需的值。但是由于 EditText 是由适配器管理的,因此单独设置文本并不能达到预期的效果。如果之后调用 e.invalidate(),文本会正确显示,if 也会跳过 adapter.notifyDatasetChanged().

所以基本上可以修改Views。但那样做是个坏主意。

相反,您应该更改数据 class。在您的情况下,您需要一种方法来为 TextInputLineBuilder 中的 EditText 设置所需的文本 (String)。然后当 getRowView() 被调用时,这个更新的文本应该与 EditText input.

一起使用

回到updateUIState():修改数据列表后调用notifyDatasetChanged()是正确的,但是不需要重新分配adapter给ListView.