Я пытаюсь создать класс, в котором определенные значения можно наблюдать, но также можно сериализовать.
Это, очевидно, работает, и сериализация работает, но она очень шаблонна-приходится добавлять сеттер для каждого отдельного поля и вручную вызывать change(...)
внутри каждого сеттера:
interface Observable {
fun change(message: String) {
println("changing $message")
}
}
@Serializable
class BlahVO : Observable {
var value2: String = ""
set(value) {
field = value
change("value2")
}
fun toJson(): String {
return Json.encodeToString(serializer(), this)
}
}
println(BlahVO().apply { value2 = "test2" })
правильно выводит
changing value2
{"value2":"test2"}
Я попытался представить делегатов:
interface Observable {
fun change(message: String) {
println("changing $message")
}
@Suppress("ClassName")
class default<T>(defaultValue: T) {
private var value: T = defaultValue
operator fun getValue(observable: Observable, property: KProperty<*>): T {
return value
}
operator fun setValue(observable: Observable, property: KProperty<*>, value: T) {
this.value = value
observable.change(property.name)
}
}
}
@Serializable
class BlahVO : Observable {
var value1: String by Observable.default("value1")
fun toJson(): String {
return Json.encodeToString(serializer(), this)
}
}
println(BlahVO().apply { value1 = "test1" })
правильно запускает обнаружение изменений, но оно не сериализуется:
changing value1
{}
Если я перейду от наблюдаемого к ReadWriteProperty,
interface Observable {
fun change(message: String) {
println("changing $message")
}
fun <T> look(defaultValue: T): ReadWriteProperty<Observable, T> {
return OP(defaultValue, this)
}
class OP<T>(defaultValue: T, val observable: Observable) : ObservableProperty<T>(defaultValue) {
override fun setValue(thisRef: Any?, property: KProperty<*>, value: T) {
super.setValue(thisRef, property, value)
observable.change("blah!")
}
}
}
@Serializable
class BlahVO : Observable {
var value3: String by this.look("value3")
fun toJson(): String {
return Json.encodeToString(serializer(), this)
}
}
результат тот же:
changing blah!
{}
Аналогично для делегатов.
var value4: String by Delegates.vetoable("value4", {
property: KProperty<*>, oldstring: String, newString: String ->
this.change(property.name)
true
})
выходы:
changing value4
{}
Делегаты, похоже, просто не работают с сериализацией Kotlin
Какие еще существуют варианты наблюдения за изменениями свойства без нарушения его сериализации, которые также будут работать на других платформах (KotlinJS, KotlinJVM, Android,...)?