数据绑定+ LiveData不适用于复杂的嵌套对象

如何解决数据绑定+ LiveData不适用于复杂的嵌套对象

我在LiveData和数据绑定库中遇到了一些意外行为。 我已经在此答案Producing HTML Tutorial中实现了CustomLiveData,因此我只能在父类中调用notifyChange()来更新UI。

我有父对象(为简便起见,一些方法被省略了):

class Day(val tasks: MutableList<RunningTask>,state: DayState = DayState.WAITING,var dayStartTime: Long = 0L,currentTaskPos: Int = 0): BaseObservable() {

    var state: DayState = state
        set(value) {
            field = value
            notifyChange()
        }

    var currentTaskPos: Int = currentTaskPos
        set(value) {
            field = value
            notifyChange()
        }

    fun start() {
        dayStartTime = System.currentTimeMillis()
        state = DayState.ACTIVE
        resetTasks()
        tasks[currentTaskPos].start()

        notifyChange()
    }
}

子对象:

class RunningTask(
    startTime: Long,var name: String = "",private val originalDuration: Long = 0L,val sound: String
): BaseObservable() {
    var startTime: Long = startTime
        set(value) {
            field = value
            uiStartTime = convertMillisToStringFormat(value)
        }

    @Bindable
    var uiStartTime: String = convertMillisToStringFormat(startTime)
        set(value) {
            field = value
            notifyPropertyChanged(BR.uiStartTime)
        }

    var duration: Long = originalDuration
        set(value) {
            field = value
        }

    var state: State = State.WAITING

    var progress: Long = 0L
        set(value) {
            field = value
        }

    var timePaused: Long = 0L
    var timeRemain: String = convertMillisToStringFormat(duration)

    enum class State {
        WAITING,ACTIVE,COMPLETED,DISABLED
    }

    fun start() {
        state = State.ACTIVE
    }
}

问题是,当我在Day的 tasks 字段中更改项目时,例如item.main_screen_task.xml的数据绑定没有更新。调用方法 start(),但是其他字段(例如 state )确实可以正确更新,所以我想问题出在里面的列表了。

fragment_main_screen.xml,在Day类字段任务中填充了recyclerview:

<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    xmlns:android="http://schemas.android.com/apk/res/android">

    <data>
        <import type="android.view.View"/>

        <variable
            name="viewmodel"
            type="com.sillyapps.meantime.ui.mainscreen.MainViewModel" />
    </data>

    <androidx.constraintlayout.widget.ConstraintLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:background="@color/colorPrimary">

        <androidx.recyclerview.widget.RecyclerView
            android:id="@+id/tasks"
            android:layout_width="0dp"
            android:layout_height="0dp"
            android:layout_marginStart="16dp"
            android:layout_marginEnd="16dp"
            android:visibility="@{viewmodel.noTemplate ? View.GONE : View.VISIBLE}"
            app:layoutManager="androidx.recyclerview.widget.LinearLayoutManager"
            app:layout_constraintBottom_toTopOf="@+id/play_button"
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintTop_toBottomOf="@+id/constraintLayout"
            tools:listitem="@layout/item_main_screen_task"
            tools:visibility="visible" />

    </androidx.constraintlayout.widget.ConstraintLayout>

</layout>

item_main_screen_task, taskState 属性基本上只是BindingAdapter,它根据Day的 state 枚举设置背景可绘制:

<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    xmlns:android="http://schemas.android.com/apk/res/android">

    <data>
        <variable
            name="task"
            type="com.sillyapps.meantime.data.RunningTask" />

        <variable
            name="taskAdapterPosition"
            type="Integer" />

        <variable
            name="clickListener"
            type="com.sillyapps.meantime.ui.ItemClickListener" />
    </data>

    <androidx.constraintlayout.widget.ConstraintLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="16dp"
        app:taskState="@{task.state}"

        android:onClick="@{() -> clickListener.onClickItem(taskAdapterPosition)}">

        <TextView
            android:id="@+id/time"
            style="@style/TimeItemStyle"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"

            android:layout_marginStart="8dp"
            android:layout_marginTop="8dp"
            android:layout_marginBottom="8dp"
            android:text="@{task.uiStartTime}"

            app:layout_constraintBottom_toBottomOf="parent"
            app:layout_constraintEnd_toStartOf="@+id/enter_name"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintTop_toTopOf="parent"
            tools:text="17:00" />

        <TextView
            android:id="@+id/enter_name"
            android:layout_width="0dp"
            android:layout_height="wrap_content"

            android:layout_marginStart="8dp"
            android:layout_marginEnd="8dp"

            android:text="@{task.name}"
            app:layout_constraintBottom_toBottomOf="@+id/time"
            app:layout_constraintEnd_toStartOf="@+id/progress"
            app:layout_constraintStart_toEndOf="@+id/time"
            app:layout_constraintTop_toTopOf="@+id/time"
            tools:text="Свободное время" />

        <TextView
            android:id="@+id/progress"
            style="@style/TimeItemStyle"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"

            android:layout_marginEnd="8dp"

            android:text="@{task.timeRemain}"
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintStart_toEndOf="@+id/enter_name"
            app:layout_constraintTop_toTopOf="@+id/time"
            tools:text="01:00" />

    </androidx.constraintlayout.widget.ConstraintLayout>

</layout>

谢谢。

解决方法

事实证明,该解决方案非常简单,但有些出乎意料

子类应该扩展BaseObservable,并在每个数据绑定字段的设置器上调用notifyChange(),诸如此类:

class RunningTask(
    startTime: Long,var name: String = "",private val originalDuration: Long = 0L,val sound: String
): BaseObservable() {
    var startTime: Long = startTime
        set(value) {
            field = value
            uiStartTime = convertMillisToStringFormat(value)
        }

    @Bindable
    var uiStartTime: String = convertMillisToStringFormat(startTime)
        set(value) {
            field = value
            notifyPropertyChanged(BR.uiStartTime)
        }

    var state: State = State.WAITING
        set(value) {
             field = value
             notifyChange()
        }
    
    ...
}

在提出问题之前,我似乎已经在uiStartTime中实现了该功能,但是我只是不知道它起作用的确切原因

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


依赖报错 idea导入项目后依赖报错,解决方案:https://blog.csdn.net/weixin_42420249/article/details/81191861 依赖版本报错:更换其他版本 无法下载依赖可参考:https://blog.csdn.net/weixin_42628809/a
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下 2021-12-03 13:33:33.927 ERROR 7228 [ main] o.s.b.d.LoggingFailureAnalysisReporter : *************************** APPL
错误1:gradle项目控制台输出为乱码 # 解决方案:https://blog.csdn.net/weixin_43501566/article/details/112482302 # 在gradle-wrapper.properties 添加以下内容 org.gradle.jvmargs=-Df
错误还原:在查询的过程中,传入的workType为0时,该条件不起作用 &lt;select id=&quot;xxx&quot;&gt; SELECT di.id, di.name, di.work_type, di.updated... &lt;where&gt; &lt;if test=&qu
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct redisServer’没有名为‘server_cpulist’的成员 redisSetCpuAffinity(server.server_cpulist); ^ server.c: 在函数‘hasActiveC
解决方案1 1、改项目中.idea/workspace.xml配置文件,增加dynamic.classpath参数 2、搜索PropertiesComponent,添加如下 &lt;property name=&quot;dynamic.classpath&quot; value=&quot;tru
删除根组件app.vue中的默认代码后报错:Module Error (from ./node_modules/eslint-loader/index.js): 解决方案:关闭ESlint代码检测,在项目根目录创建vue.config.js,在文件中添加 module.exports = { lin
查看spark默认的python版本 [root@master day27]# pyspark /home/software/spark-2.3.4-bin-hadoop2.7/conf/spark-env.sh: line 2: /usr/local/hadoop/bin/hadoop: No s
使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams[&#39;font.sans-serif&#39;] = [&#39;SimHei&#39;] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -&gt; systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping(&quot;/hires&quot;) public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate&lt;String
使用vite构建项目报错 C:\Users\ychen\work&gt;npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-