Как определить свойство делегата, которое нужно инициализировать с задержкой в ​​Kotlin? - PullRequest
0 голосов
/ 02 декабря 2018

Мне нужно определить свойство isBackgroundWindow по классу делегата PreferenceTool, но следующий код вызовет ошибку NullPointerException.

Я знаю, что private val prefs: SharedPreferences by lazy { } в PreferenceTool<T> является ленивым, поэтому объект this не инициализируется, когда система вызывает PreferenceTool(this, getString(R.string.IsBackgroundName) , false),, это приведет к нулевой ошибке.

Я надеюсь использовать код private lateinit var isBackgroundWindow: Boolean by PreferenceTool(this, getString(R.string.IsBackgroundName) , false), но модификатор 'lateinit' не разрешенделегированные свойства .

Как это сделать?

Main

class UIHome : AppCompatActivity() {   

   //I think the object this is not initialized, it will cause null error.
    private var isBackgroundWindow: Boolean by PreferenceTool(this, getString(R.string.IsBackgroundName) , false) 

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.layout_home)

        isBackgroundWindow=false
     }

}

Класс делегата

class PreferenceTool<T>(private val context: Context, private val name: String,  private val default: T) {

    private val prefs: SharedPreferences by lazy {     
        context.defaultSharedPreferences     
    }

    operator fun getValue(thisRef: Any?, property: KProperty<*>): T = findPreference(name, default)

    operator fun setValue(thisRef: Any?, property: KProperty<*>, value: T) {
        putPreference(name, value)
    }

    @Suppress("UNCHECKED_CAST")
    private fun findPreference(name: String, default: T): T = with(prefs) {
        val res: Any = when (default) {
            is Long -> getLong(name, default)
            is String -> getString(name, default)
            is Int -> getInt(name, default)
            is Boolean -> getBoolean(name, default)
            is Float -> getFloat(name, default)
            else -> throw IllegalArgumentException("This type can be saved into Preferences")
        }

        res as T
    }

    @SuppressLint("CommitPrefEdits")
    private fun putPreference(name: String, value: T) = with(prefs.edit()) {
        when (value) {
            is Long -> putLong(name, value)
            is String -> putString(name, value)
            is Int -> putInt(name, value)
            is Boolean -> putBoolean(name, value)
            is Float -> putFloat(name, value)
            else -> throw IllegalArgumentException("This type can't be saved into Preferences")
        }.apply()
    }
}

1 Ответ

0 голосов
/ 04 декабря 2018

Вы можете латинитировать переменную isBackground следующим образом. Ваш класс UIHome должен быть

 class UIHome : AppCompatActivity() {

 var isBackground: Boolean by Delegates.notNull<Boolean>()

override fun onCreate(savedInstanceState: Bundle?) {
 super.onCreate(savedInstanceState)
 setContentView(R.layout.activity_uihome)
 isBackground = isBackgroundWindow()

}

fun isBackgroundWindow(): Boolean {
val isBackgroundWindow: Boolean by PreferenceTool(
    this, getString(R.string.IsBackgroundName), false
)
return isBackgroundWindow
}
}

Это исключение нулевого указателя во избежание

isBackground?.let { 

  // your code
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...