Это можно сделать гораздо более плавно, если вы полагаетесь на наблюдаемые свойства.Создайте свойство, содержащее текущую тему, и удалите старую и добавьте новую тему при изменении этого свойства.
Для привязки к свойству активной темы вы можете использовать встроенную функциональность группы переключения.
Вот полное приложение, показывающее это:
class MyThemeApp : App(SettingsView::class) {
val themeController: ThemeController by inject()
override fun start(stage: Stage) {
super.start(stage)
// Make sure we initialize the theme selection system on start
themeController.start()
}
}
class ThemeController : Controller() {
// List of available themes
val themes = SimpleListProperty<KClass<out Stylesheet>>(listOf(LightTheme::class, DarkTheme::class).observable())
// Property holding the active theme
val activeThemeProperty = SimpleObjectProperty<KClass<out Stylesheet>>()
var activeTheme by activeThemeProperty
fun start() {
// Remove old theme, add new theme on change
activeThemeProperty.addListener { _, oldTheme, newTheme ->
oldTheme?.let { removeStylesheet(it) }
newTheme?.let { importStylesheet(it) }
}
// Activate the first theme, triggering the listener above
activeTheme = themes.first()
}
}
class SettingsView : View("Settings") {
val settings: ThemeController by inject()
override val root = form {
fieldset("Theme") {
field {
vbox {
togglegroup {
// One radio button for each theme, with their value set as the theme
settings.themes.forEach { theme ->
radiobutton(theme.simpleName, getToggleGroup(), theme)
}
// The toggle group value is bound to the activeThemeProperty
bind(settings.activeThemeProperty)
}
}
}
buttonbar {
button("OK").action(this@SettingsView::close)
}
}
}
}
// Two themes for completeness
class DarkTheme : Stylesheet() {
init {
root {
backgroundColor += Color.DARKGREEN
}
}
}
class LightTheme : Stylesheet() {
init {
root {
backgroundColor += Color.LIGHTCYAN
}
}
}