Как вложить несколько делегатов свойств в Kotlin

Я столкнулся со случаем, когда я хочу «связать» несколько делегатов (направить вывод одного в другой).

Это кажется возможным:

private val errorLogList by listSO(listOf<StateObject<Luxeption>>(), SODest.NONE, publicSOAccessRights())
val errorLog: StateObject<List<StateObject<Luxeption>>> by addToSet(errorLogList)

Однако выглядит это не слишком хорошо :). Я хотел бы сделать это в одной строке следующим образом:

val errorLog: StateObject<List<StateObject<Luxeption>>> by addToSet(
   listSO(listOf<StateObject<Luxeption>>(), SODest.NONE, publicSOAccessRights())
)

Мой вопрос: Возможен ли такой тип создания свойств с помощью делегатов в Kotlin?


Вот обе реализации моих делегатов:

добавить в набор:

open class ChildSOReturner {

    val set: Set<StateObject<*>> = setOf()

    inline fun <reified T> addToSet(so: T) = object: ReadOnlyProperty<Any?, T> {
        override operator fun getValue(thisRef: Any?, property: KProperty<*>): T {
            if (thisRef is T) {
                set.plus(so)
                return so
            } else throw IllegalArgumentException()
        }
    }
}

списокТак:

fun <O> listSO(
    initialState: List<StateObject<O>>,
    soDest: SODest,
    soAccessRights: SOAccessRights
) = object : ReadOnlyProperty<Any?, StateObject<List<StateObject<O>>>> {

    override operator fun getValue(thisRef: Any?, property: KProperty<*>): StateObject<List<StateObject<O>>> {
        val meta = SOMeta(SOId(property.name), soDest, soAccessRights)
        return StateObjectList(initialState, meta)
    }

}

person p_0g_amm3_    schedule 07.02.2020    source источник


Ответы (1)


Это оказалось довольно сложно, но возможно (если я что-то не упускаю, и это не проверено, но идея должна работать):

fun <T, U, V> composeProperties(prop: ReadOnlyProperty<T, U>, f: (U) -> ReadOnlyProperty<T, V>) : ReadOnlyProperty<T, V> {
    var props = mutableMapOf<Pair<T, KProperty<*>>, ReadOnlyProperty<T, V>>()
    return object : ReadOnlyProperty<T, V> {
        override operator fun getValue(thisRef: T, property: KProperty<*>): V {
            val prop1 = props.getOrPut(Pair(thisRef, property)) { 
                f(prop.getValue(thisRef, property))
            }
            return prop1.getValue(thisRef, property)
        }
    }
}

А затем использовать

val errorLog: ... by composeProperties(listSO(...)) { addToSet(it) }
person Alexey Romanov    schedule 15.02.2020
comment
Есть ли способ объединить composeProperties(..) { addToSet(it) } в такой вызов, как addComposed(..)? Я не могу понять это. - person p_0g_amm3_; 16.02.2020