显示列表视图后尝试以编程方式设置 LISTVIEW 内的 SWITCH 状态

2023-11-29

所以我有一个警报列表,并将其绑定到列表视图 lstAlarms。在我的自定义 ListView 布局中,我还有一个开关,我想根据警报的状态以编程方式设置它。我想在 ListView 显示后立即执行此操作。

请参阅下面我的代码。 显示ListView的方法是DisplayAlarmList()。 我试图用来设置开关状态的方法是 InitSwitches(),它是在 DisplayAlarmList() 内部调用的。 DisplayAlarmList() 在 onCreate() 方法中调用。

public void DisplayAlarmList()
{
    final String[] columns = {Database.getAlarmID(), Database.getAlarmTime(), Database.getAlarmName(), Database.getAlarmStatus(), Database.getAlarmRepeats()};
    Cursor c = Database.selectAlarm(db, Database.getTableName(), columns, null, null, null, null, null);

    int[] to = new int[]{
            R.id.alarmID,
            R.id.alarmTime,
            R.id.alarmName,
            R.id.alarmStatus,
            R.id.alarmRepeats,
    };

    SimpleCursorAdapter ca = new SimpleCursorAdapter(this,
            R.layout.alarm_info,
            c,
            columns,
            to,
            0);

    lstAlarm.setAdapter(ca);
    InitSwitches();

    lstAlarm.setOnItemClickListener(new OnItemClickListener()
    {

        @Override
        public void onItemClick(AdapterView<?> listView, View view, int position, long id)
        {
            Alarm alarm = new Alarm();
            Cursor selectedCursor = (Cursor) listView.getItemAtPosition(position);

            Switch s = (Switch) view.findViewById(R.id.alarmSwitch);

            String whereArgs = Integer.toString(selectedCursor.getInt(selectedCursor.getColumnIndexOrThrow(Database.getAlarmID())));


            Cursor data = Database.RawQuery(db, "SELECT * FROM " + Database.getTableName() + " WHERE " + Database.getAlarmID() + " = " + whereArgs);

            if (data.moveToFirst())
            {
                alarm.setAlarmID(data.getString(data.getColumnIndexOrThrow(Database.getAlarmID())));
                alarm.setAlarmName(data.getString(data.getColumnIndexOrThrow(Database.getAlarmName())));
                alarm.setAlarmTime(data.getString(data.getColumnIndexOrThrow(Database.getAlarmTime())));
                alarm.setAlarmSound(data.getString(data.getColumnIndexOrThrow(Database.getAlarmSound())));
                alarm.setAlarmRepeats(data.getString(data.getColumnIndexOrThrow(Database.getAlarmRepeats())));
                alarm.setAlarmStatus(data.getString(data.getColumnIndexOrThrow(Database.getAlarmStatus())));
            }

            Intent i = new Intent(view.getContext(), ScreenEdit.class);
            i.putExtra("editAlarm", new Gson().toJson(alarm));
            startActivityForResult(i, EDIT_ALARM);

        }
    });
}

public void InitSwitches()
{
    View v;
    Switch s;
    int index = 0;
    String query = "Select " + Database.getAlarmID() + ", " + Database.getAlarmStatus() + " FROM " + Database.getTableName();
    Cursor statuses = Database.RawQuery(db, query);

    while (statuses.moveToNext())
    {
        v = lstAlarm.getAdapter().getView(index++, null, null);
        s = (Switch) v.findViewById(R.id.alarmSwitch);

        if (statuses.getString(statuses.getColumnIndexOrThrow(Database.getAlarmStatus())).equals("ON")){
            s.toggle();
        }
        else{                

        }
    }
}

应用程序的屏幕截图

正如您从屏幕截图中看到的,第二行中的开关应该是打开的,但事实并非如此。我也尝试过 InvilidateView() 但没有成功。请帮忙。


问题是 ListView 是不断重绘的视图之一,这意味着这很糟糕,因为重绘时它会丢失状态

if (statuses.getString(statuses.getColumnIndexOrThrow(Database.getAlarmStatus())).equals("ON")){
    s.toggle();
}

首先创建报警项……你应该根据你的需要来制作。这是我的

public class AlarmItem {

    private String alarmName;
    private String alarmDescption;
    private boolean state;
    private long id;

    public AlarmItem(String alarmName, String alarmDescption, long id, boolean state) {
        this.alarmName = alarmName;
        this.alarmDescption = alarmDescption;
        this.id = id;
        this.state = state;
    }

    public long getId() {
        return id;
    }

    public void setId(long id) {
        this.id = id;
    }

    public String getAlarmName() {
        return alarmName;
    }

    public void setAlarmName(String alarmName) {
        this.alarmName = alarmName;
    }

    public String getAlarmDescption() {
        return alarmDescption;
    }

    public void setAlarmDescption(String alarmDescption) {
        this.alarmDescption = alarmDescption;
    }

    public boolean getState() {
        return state;
    }

    public void setState(boolean state) {
        this.state = state;
    }
}

现在我们需要一个自定义开关类因为这个:LINK

import android.content.Context;
import android.util.AttributeSet;
import android.widget.Switch;

public class CustomSwitch extends Switch {


    private OnCheckedChangeListener mListener;

    public CustomSwitch(Context context) {
        super(context);
    }

    public CustomSwitch(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public CustomSwitch(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }

    public CustomSwitch(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
        super(context, attrs, defStyleAttr, defStyleRes);
    }

    @Override
    public void setOnCheckedChangeListener(OnCheckedChangeListener listener) {
        // Do not call supper method
        mListener = listener;
    }

    @Override
    public void setChecked(boolean checked) {
        super.setChecked(checked);

        if (mListener != null) {
            mListener.onCheckedChanged(this, checked);
        }
    }

    public void setCheckedProgrammatically(boolean checked) {
        // You can call super method, it doesn't have a listener... he he :)
        super.setChecked(checked);
    }
}

创建一个布局文件并命名Alarm_list_item.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
                android:layout_width="match_parent"
                android:layout_height="50dp"
                android:orientation="vertical">

    <com.example.alarm.list.CustomSwitch
        android:id="@+id/alarmSwitch"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentRight="true"
        android:layout_centerVertical="true"
        android:layout_marginRight="10dp"/>

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginLeft="5dp"
        android:layout_toLeftOf="@id/alarmSwitch"
        android:orientation="vertical">

        <TextView
            android:id="@+id/tvAlarmName"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginTop="5dp"
            android:text="Alarm Name"
            android:textSize="18sp"/>

        <TextView
            android:id="@+id/tvAlarmDesc"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginTop="5dp"
            android:text="Alarm description"
            android:textSize="15sp"/>
    </LinearLayout>


</RelativeLayout>

现在主要布局 ->活动_main.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:layout_width="match_parent"
              android:layout_height="match_parent"
              android:orientation="vertical">

    <ListView
        android:id="@+id/lvAlarms"
        android:layout_width="match_parent"
        android:layout_height="match_parent"/>

</LinearLayout>

现在我们需要一个适配器来负责绘制项目及其处理。为班级命名报警适配器

import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.CompoundButton;
import android.widget.TextView;

import java.util.List;

public class AlarmAdapter extends BaseAdapter {

    private List<AlarmItem> listOfAlarms;
    private Context context;
    private OnAlarmCheckedChangeListener mCallback;
    // avoid constant allocation
    private View tmpView;
    private AlarmItemViewHolder mHolder;
    private AlarmItem tmpItem;
    public AlarmAdapter(List<AlarmItem> listOfAlarms, Context context, OnAlarmCheckedChangeListener callBack) {
        this.listOfAlarms = listOfAlarms;
        this.context = context;
        mCallback = callBack;
    }

    @Override
    public int getCount() {
        return listOfAlarms == null ? 0 : listOfAlarms.size();
    }

    @Override
    public AlarmItem getItem(int i) {
        return listOfAlarms == null ? null : listOfAlarms.get(i);
    }

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

    @Override
    public View getView(final int i, View view, ViewGroup viewGroup) {

        tmpItem = listOfAlarms.get(i);

        if (view == null) {
            LayoutInflater inflater = LayoutInflater.from(context);
            tmpView = inflater.inflate(R.layout.alarm_list_item, null, false);
            mHolder = new AlarmItemViewHolder(tmpView);
            tmpView.setTag(mHolder);
        }
        else {
            tmpView = view;
            mHolder = (AlarmItemViewHolder) view.getTag();
        }

        mHolder.getAlarmNameTextView().setText(tmpItem.getAlarmName());
        mHolder.getAlarmDescriptionTextView().setText(tmpItem.getAlarmDescption());
        mHolder.getSwitch().setCheckedProgrammatically(tmpItem.getState());

        mHolder.getSwitch().setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
            @Override
            public void onCheckedChanged(CompoundButton compoundButton, boolean b) {
                listOfAlarms.get(i).setState(b);
                mCallback.onAlarmStateChanged(listOfAlarms.get(i), i);
            }
        });


        return tmpView;
    }

    public void clear() {
        listOfAlarms.clear();
        notifyDataSetChanged();
    }

    public void refill(List<AlarmItem> listOfAlarms) {
        this.listOfAlarms = listOfAlarms;
        notifyDataSetChanged();
    }

    public void toggleAllSwitches() {
        for (AlarmItem item : listOfAlarms) {
            item.setState(!item.getState());
        }
        notifyDataSetChanged();
    }

    public interface OnAlarmCheckedChangeListener {
        public void onAlarmStateChanged(AlarmItem item, int postionInList);
    }

    private class AlarmItemViewHolder {

        View base;
        CustomSwitch mSwitch;
        TextView mAlarmName;
        TextView mAlarmDescription;

        public AlarmItemViewHolder(View base) {
            this.base = base;
        }

        public CustomSwitch getSwitch() {
            if (mSwitch == null) {
                mSwitch = (CustomSwitch) base.findViewById(R.id.alarmSwitch);
            }
            return mSwitch;
        }

        public TextView getAlarmNameTextView() {
            if (mAlarmName == null) {
                mAlarmName = (TextView) base.findViewById(R.id.tvAlarmName);
            }
            return mAlarmName;
        }

        public TextView getAlarmDescriptionTextView() {
            if (mAlarmDescription == null) {
                mAlarmDescription = (TextView) base.findViewById(R.id.tvAlarmDesc);
            }
            return mAlarmDescription;
        }
    }
}

现在终于主要活动

import android.app.ActionBar;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.support.v7.app.ActionBarActivity;
import android.view.View;
import android.widget.AdapterView;
import android.widget.ListView;
import android.widget.Toast;

import java.util.ArrayList;
import java.util.List;


public class MainActivity extends ActionBarActivity implements AdapterView.OnItemClickListener, AlarmAdapter.OnAlarmCheckedChangeListener {

    private ListView listView;
    private AlarmAdapter adapter;
    private Toast toast;
    private Handler handler;
    private Runnable handlerRunnable;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);


        // Make a nice actionBar title
        ActionBar ab = getActionBar();
        if (ab != null) {
            ab.setTitle("Alarm List");
        }

        listView = (ListView) findViewById(R.id.lvAlarms);

        // Simulating alarms from database. You need to convert your items to these
        List<AlarmItem> alarmsFromDb = new ArrayList<>();
        alarmsFromDb.add(new AlarmItem("Alarm 1", "Lalalala", 1, true));
        alarmsFromDb.add(new AlarmItem("Alarm 2", "something", 2, false));
        alarmsFromDb.add(new AlarmItem("Alarm 3", "gfdgdf", 3, true));
        alarmsFromDb.add(new AlarmItem("Alarm 4", "sda", 4, true));
        alarmsFromDb.add(new AlarmItem("Alarm 5", "yxcxyc", 5, false));
        alarmsFromDb.add(new AlarmItem("Alarm 6", "dsfsd", 6, false));

        adapter = new AlarmAdapter(alarmsFromDb, this, this);

        listView.setAdapter(adapter);
        listView.setOnItemClickListener(this);

        // Toggle all switches after 5s... this is what you need, right?
        handlerRunnable = new Runnable() {
            @Override
            public void run() {
                adapter.toggleAllSwitches();
                showToast("All switches toggeled :)");
            }
        };

        handler = new Handler(Looper.getMainLooper());
        handler.postDelayed(handlerRunnable, 5 * 1000);
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        if (handler != null) {
            handler.removeCallbacks(handlerRunnable);
            handler = null;
            handlerRunnable = null;
        }
    }

    private void showToast(String str) {
        if (toast != null) {
            toast.cancel();
        }
        toast = Toast.makeText(this, str, Toast.LENGTH_SHORT);
        toast.show();
    }

    @Override
    public void onAlarmStateChanged(AlarmItem item, int postionInList) {
        String onOff = item.getState() ? "ON" : "OFF";
        showToast("Alarm " + item.getAlarmName() + " is: " + onOff);
    }

    @Override
    public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
        AlarmItem item = adapter.getItem(position);
        showToast("Alarm " + item.getAlarmName() + " clicked");
    }
}

这是完整的android studio项目,如果您有任何问题:LINK

现在您需要做的就是将数据库中的结果转换为此列表或修改报警项和用户界面。该解决方案将起作用,并且您已经在适配器中拥有一些有用的方法。

快乐编码!

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

显示列表视图后尝试以编程方式设置 LISTVIEW 内的 SWITCH 状态 的相关文章

  • 意图?如何注册以获得传入电子邮件提醒?

    这里是极限 Android 开发新手 好吧 Android 开发新手 而不是一般开发 我想编写一个应用程序 当电子邮件进入设备时会收到通知 并根据电子邮件中的数据 主题 发件人 收件人等 采取各种操作 我认为我要做的是创建一个带有 Inte
  • 如何使用retrofit2进行GET请求?

    我有一个在本地主机上运行的安静的 Web 服务 我想在该剩余 URL 上发出 Retrofit2 GET 请求 MainActivity java private void requestData public static final S
  • Android向后兼容技术

    我现在在开发基于最新 API 15 ICS 的 15 项活动 Android 应用程序方面取得了进展 现在我发现应用程序的主要功能主义者即使支持 android v4 也不向后兼容 例如 1 fragment事务动画 2 将StringSe
  • Android 中的 java.util.Observable 是线程安全的吗?

    Android 中的 java util Observable 是线程安全的吗 这文档 http developer android com reference java util Observable html说只有deleteObser
  • ListView:防止视图回收

    我有一个使用回收视图的 ListView 我试图阻止视图被回收 所以我使用 setHasTransientState android support v4 view ViewCompatJB setHasTransientState Vie
  • 如何忽略 LeakCanary 中的某些类?

    有人能给我一个如何忽略 LeakCanary 中的某些类的有效示例吗 我正在查看这个示例 以忽略 LeakCanary 中第三方库中的某些类 但我不知道将其放在应用程序中的何处 我把它放在我的应用程序类中 但这些变量和方法有错误 isInA
  • 我的 Android 设备需要安装哪个驱动程序才能运行我的应用程序?

    我购买了 intex mobile 来在真实设备中测试我的 Android 应用程序 然而 该设备不存在于 OEM USB 驱动程序列表中 android 提供的设备列表中 我检查了 intex 官方网站 但不确定到底需要安装哪个驱动程序
  • Android 谷歌地图 V2 已停止

    我正在尝试构建地图应用程序并关注这个链接 https blog emildesign rhcloud com p 435一步步 我在这里找到了类似的主题 但对我没有帮助 我想显示地图 但是当我运行它时 它返回强制关闭和我的 Android
  • Android BLE 扫描在后台几分钟后停止

    当我为公司开发新冠肺炎接触者追踪应用程序时 我在后台遇到了 Android 扫描停止问题 这是我尝试过的 添加前台服务 禁用手机中所有与电池相关的优化选项 启用后台运行的应用程序 测试设备 搭载 Android 10 的 Galaxy S2
  • 收到“无法解析上传的APK的AndroidManifest.xml。它是否正确编译?”启用 Google 应用签名后出现错误

    启用后谷歌应用程序签名 https support google com googleplay android developer answer 7384423 hl en 每次我尝试将签名版本 APK 上传到 Play 商店时 都会收到一
  • 如何在Android网格视图中设置单元格大小?

    我正在尝试为应用程序制作一个带有大图标的网格视图 但我找不到任何有关修改 Android 上网格布局上的单元格大小的教程 有人可以给我一个例子或相关链接吗 Thanks 就像另一个一样适配器视图 http developer android
  • Android Drawable 绘图性能?

    在我看来 我有一个简单的 ARGB 可绘制对象 大约需要 2 毫秒才能绘制 但我可以在 0 5 毫秒内绘制与位图相同的文件 只是一些快速代码 我真的不能认为它是一个选项 优化可绘制对象的绘制速度的最佳方法是什么 这取决于可绘制的数量以及每个
  • 有没有办法在多个嵌套的 RecyclerView 之间共享同一个 LayoutManager

    我正在开发一个显示游戏列表的应用程序 在每个游戏的 itemView 内 我还有一个要显示的视频列表 预览和结构如下 我部署了一个RecyclerView作为窗口根视图 然后对于视频 我使用网格样式的RecyclerView来显示 所以这里
  • 如何更改 Android 12 启动屏幕中的图标形状?

    我想要矩形形状的启动屏幕图标 而不是 android 12 中的圆形形状 我不相信你可以 如果你看这里的第 3 点 https developer android com about versions 12 features splash
  • 无法登录 Google Play 游戏服务

    我在开发者控制台上使用包名称和正确的签名证书设置了我的游戏 并为其创建了排行榜 但没有创建任何成就 然后 我从以下位置下载了示例 Type A Number Challenge 和 BaseGameUtils https developer
  • 从Android客户端登录appengine

    我正在尝试登录应用程序引擎并访问应用程序引擎中的用户服务API 基本上我希望能够看到谁登录了我的 servlet 我正在使用从 android 获取 authtoken 然后从应用程序引擎获取 ASID 或 SACID cookie 的身份
  • 如何在 onDraw() 方法中定义与像素无关的高度

    我扩展了 View 来构建自定义小部件 我想用独立的像素单位定义小部件的高度 我认为可以通过将像素密度乘以所需的高度来完成 但我不知道该怎么做 到目前为止我所拥有的 最小化 public class Timeline extends Vie
  • Android 4.2 - Environment.getExternalStorageDirectory().getPath() 行为

    我一直在开发一个android应用程序 在上次更新到4 2之前 我使用 Environment getExternalStorageDirectory getPath 它返回了我 storage sdcard0 但自从更新后我现在得到了 s
  • 改造方法调用可能会产生“java.lang.NullPointerException”

    使用 Retrofit 2 3 0 我在 Android Studio 中收到以下消息 有关如何删除此 IDE 错误消息的任何建议 谢谢 来自Response文档 http square github io retrofit 2 x ret
  • Application.onLowMemory() 未调用

    我创建了自己的应用程序类 我尝试调试它 代码在 Application onCreate 处停止 但不会在 onLowMemory 处停止 为了测试该场景 我打开了许多其他高内存应用程序 我看到的是调试会话终止 在 Eclipse 中 并且

随机推荐