TornadoFX编程指南,第6章,类型安全CSS

译自《Type-Safe CSS

类型安全CSS

虽然您可以在JavaFX中创建纯文本的CSS样式表,但TornadoFX提供了将类型安全性和编译的CSS引入JavaFX的选项。 您可以方便地选择在自己的类中创建样式(create styles in its own class),或者在控件声明中内联(inline within a control declaration)。

内联CSS

最快速地且最简单地给控件指定样式的方式是,调用一个给定的Node的内联style { }函数。 给定控件上可用的所有CSS属性都可以以类型安全的方式提供,具有编译检查(compilation checks)和自动完成(auto-completion)。

例如,您可以给Button的边框(使用box()函数)进行样式化,粗体显示它的字体,然后旋转它(图6.1)。

button("Press Me") {
    style {
        fontWeight = FontWeight.EXTRA_BOLD
        borderColor += box(
                top = Color.RED,
                right = Color.DARKGREEN,
                left = Color.ORANGE,
                bottom = Color.PURPLE
        )
        rotate = 45.deg
    }

    setOnAction { println("You pressed the button") }
}
图6.1

当您希望在不破坏Button的声明流的情况下调整控件时,这是特别有用的。 但是,请记住, style { }将替换应用于该控件的所有样式,除非您为其可选的append参数传递true

style(append = true) {
      ....
}

有时您想一次性将相同的样式应用于许多节点。style { }函数也可以应用于包含节点的任何Iterable

vbox {
    label("First")
    label("Second")
    label("Third")
    children.style {
        fontWeight = FontWeight.BOLD
    }
}

fontWeight样式适用于vbox的所有子项,本质上就是我们添加的所有标签。

当您的样式复杂度超过一定阈值时,您可能需要考虑使用我们将在下面介绍的样式表(Stylesheets)。

使用样式表应用样式类(Applying Style Classes with Stylesheets)

如果要组织(organize),重用(re-use),组合(combine)和覆盖(override)样式,您需要使用Stylesheet 。 传统上在JavaFX中,样式表在项目中包含的纯CSS文本文件中定义。 但是,TornadoFX允许使用纯Kotlin代码创建样式表。 这具有编译检查,自动完成和其他带有静态类型代码的好处(compilation checks, auto-completion, and other perks)。

要声明Stylesheet,将其扩展到您自己的类以保存您的自定义样式。

import tornadofx.*

class MyStyle: Stylesheet() {
}

接下来,您将要指定其companion object来保存可以轻松检索的类级属性。 声明一个新的cssclass() 代理属性,名为tackyButton,并定义我们将用于其边框的四种颜色。

import javafx.scene.paint.Color
import tornadofx.*

class MyStyle: Stylesheet() {

    companion object {
        val tackyButton by cssclass()

        private val topColor = Color.RED
        private val rightColor = Color.DARKGREEN
        private val leftColor = Color.ORANGE
        private val bottomColor = Color.PURPLE
    }
}

注意,您也可以使用c()函数使用RGB值或颜色字符串快速构建颜色。

  private val topColor = c("#FF0000")
  private val rightColor = c("#006400")
  private val leftColor = c("#FFA500")
  private val bottomColor = c("#800080")

最后,声明一个init()块来将样式应用于类。 定义您的选择(selection),并提供一个操纵其各种属性的块。 (对于复合选择,调用s()函数,它是select()函数的别名)。 设置rotate到10度,使用四种颜色和box()函数定义borderColor ,使字体系列 “Comic Sans MS”,并将fontSize增加到20像素。 请注意, Number类型的扩展属性可以快速生成该单位的值,例如10度为10.deg,20像素为20.px

import javafx.scene.paint.Color
import tornadofx.*

class MyStyle: Stylesheet() {

    companion object {
        val tackyButton by cssclass()

        private val topColor = Color.RED
        private val rightColor = Color.DARKGREEN
        private val leftColor = Color.ORANGE
        private val bottomColor = Color.PURPLE
    }

    init {
        tackyButton {
            rotate = 10.deg
            borderColor += box(topColor,rightColor,bottomColor,leftColor)
            fontFamily = "Comic Sans MS"
            fontSize = 20.px
        }
    }
}

现在,您可以将tackyButton样式应用于支持这些属性的按钮,标签和其他控件。 虽然此样式可以与其他控件(如标签)配合使用,但我们将在此示例中定位按钮。

首先,将MyStyle样式表加载到应用程序中。

class MyApp: App(MyView::class, MyStyle::class) {
    init {
        reloadStylesheetsOnFocus()
    }
}

reloadStylesheetsOnFocus()函数调用将指示TornadoFX每次Stage获取焦点时重新加载样式表。 您还可以将--live-stylesheets参数传递给应用程序来完成此操作。

重要信息:要reload可以工作,您必须以调试模式运行JVM,并且必须先指示IDE重新编译,然后再切换回应用程序。 没有这些步骤,什么都不会发生。 这也适用于类似的reloadViewsOnFocus(),但重新加载整个视图,而不仅仅是样式表。 这样,您可以在 “代码更改,编译,刷新(code change, compile, refresh)” 的方式中快速演变UI。

您可以通过调用其addClass()函数将样式直接应用于控件。 为两个按钮提供MyStyle.tackyButton样式的代码如下(图6.2)。

class MyView: View() {
    override val root = vbox {
        button("Press Me") {
            addClass(MyStyle.tackyButton)
        }
        button("Press Me Too") {
            addClass(MyStyle.tackyButton)
        }
    }
}
图6.2

Intellij IDEA可以执行一个quickfix导入成员变量,允许addClass(MyStyle.tackyButton)缩短为addClass(tackyButton),如果你愿意的话。

您也可以使用removeClass()来删除指定的样式。

将样式定位到类型(Targeting Styles to a Type)

使用纯Kotlin的好处之一是您可以使用Kotlin代码来严格的操纵UI控件的行为和条件。 例如,您可以通过遍历控件的children,过滤仅针对Buttons的子项,并向其中应用addClass(),将样式应用于任何Button

class MyView: View() {
    override val root = vbox {
        button("Press Me")
        button("Press Me Too")

        children.asSequence()
                .filter { it is Button }
                .forEach { it.addClass(MyStyle.tackyButton) }
    }
}

事实上,一次操作几个节点上的类是很常见的,因此TornadoFX为它提供了一个快捷方式:

children.filter { it is Button }.addClass(MyStyle.tackyButton) }

您还可以通过选择和修改Stylesheetbutton来定位应用程序中的所有Button实例。 这将应用样式给所有按钮。

import javafx.scene.paint.Color
import tornadofx.*

class MyStyle: Stylesheet() {

    companion object {
        val tackyButton by cssclass()

        private val topColor = Color.RED
        private val rightColor = Color.DARKGREEN
        private val leftColor = Color.ORANGE
        private val bottomColor = Color.PURPLE
    }

    init {
        button {
            rotate = 10.deg
            borderColor += box(topColor,rightColor,leftColor,bottomColor)
            fontFamily = "Comic Sans MS"
            fontSize = 20.px
        }
    }
}
import javafx.scene.layout.VBox
import tornadofx.*

class MyApp: App(MyView::class, MyStyle::class) {
    init {
        reloadStylesheetsOnFocus()
    }
}
class MyView: View() {
    override val root = vbox {
        button("Press Me")
        button("Press Me Too")
    }
}
图6.3

还可以选择多个类和控件类型来混合并匹配样式(mix-and-match)。 例如,您可以将标签和按钮的字体大小设置为20像素,并为按钮创建粘性边框(tacky borders)和字体(图6.4)。

class MyStyle: Stylesheet() {

    companion object {

        private val topColor = Color.RED
        private val rightColor = Color.DARKGREEN
        private val leftColor = Color.ORANGE
        private val bottomColor = Color.PURPLE
    }

    init {
        s(button, label) {
            fontSize = 20.px
        }
        button {
            rotate = 10.deg
            borderColor += box(topColor,rightColor,leftColor,bottomColor)
            fontFamily = "Comic Sans MS"
        }
    }
}
class MyApp: App(MyView::class, MyStyle::class) {
    init {
        reloadStylesheetsOnFocus()
    }
}

class MyView: View() {
    override val root = vbox {
        label("Lorem Ipsum")
        button("Press Me")
        button("Press Me Too")
    }
}
图6.4

多值CSS属性(Multi-Value CSS Properties)

某些CSS属性可以接受多个值,而TornadoFX样式表可以使用multi()函数来简化。 这允许您通过varargs参数指定多个值,并让TornadoFX处理其余值。 例如,您可以将多个背景颜色和插图嵌入到控件中(图6.5)。

label("Lore Ipsum") {
    style {
        fontSize = 30.px
        backgroundColor = multi(Color.RED, Color.BLUE, Color.YELLOW)
        backgroundInsets = multi(box(4.px), box(8.px), box(12.px))
    }
}
图6.5

multi()函数应该在接受多个值的地方工作。 如果您只需要为接受多个值的属性分配一个值,则需要使用plusAssign()运算符来添加它(图6.6)。

label("Lore Ipsum") {
    style {
        fontSize = 30.px
        backgroundColor += Color.RED
        backgroundInsets += box(4.px)
    }
}
图6.6

嵌套样式(Nesting Styles)

在选择器块(selector block)中,您可以应用更多的针对子控件的样式。

例如,定义一个名为critical的CSS类。 使其在任何适用于它的控件上放置一个橙色边框,并将其pad设为5像素。

class MyStyle: Stylesheet() {

    companion object {
        val critical by cssclass()
    }

    init {
        critical {
            borderColor += box(Color.ORANGE)
            padding = box(5.px)
        }
    }
}

但是假设当我们对任何控件(例如HBox应用critical时,我们希望它可以对控件内的按钮添加额外的样式。 嵌套另一个选择(Nesting another selection)将会做到这一点。

class MyStyle: Stylesheet() {
    companion object {
        val critical by cssclass()
    }
    init {
        critical {
            borderColor += box(Color.ORANGE)
            padding = box(5.px)
            button {
                backgroundColor += Color.RED
                textFill = Color.WHITE
            }
        }## Targeting Control Types
         ## If you want to style controls by their type, you can save yourself the effort
    }
}

现在当你应用critical时候,一个HBoxHBox里面的所有按钮都会得到button定义样式(图6.7)

class MyApp: App(MyView::class, MyStyle::class) {
    init {
        reloadStylesheetsOnFocus()
    }
}

class MyView: View() {
    override val root = hbox {
        addClass(MyStyle.critical)
        button("Warning!")
        button("Danger!")
    }
}
图6.7

在这里不要混淆一个关键的事情。 这个橙色边框只适用于HBox,因为它已经应用到critical。 按钮没有橙色边框,因为他们是HBox的子节点。 虽然他们的风格是由critical定义的,但它们不会继承其父级的样式,只能为button定义。

如果您希望按钮也可以获得橙色边框,则需要将critical类直接应用于它们。 您将要使用and()将特定样式应用于也被声明为critical的按钮。

class MyStyle: Stylesheet() {

    companion object {
        val critical by cssclass()
    }

    init {
        critical {

            borderColor += box(Color.ORANGE)
            padding = box(5.px)

            and(button) {
                backgroundColor += Color.RED
                textFill = Color.WHITE
            }
        }
    }
}
class MyApp: App(MyView::class, MyStyle::class) {
    init {
        reloadStylesheetsOnFocus()
    }
}

class MyView: View() {
    override val root = hbox {
        addClass(MyStyle.critical)

        button("Warning!") {
            addClass(MyStyle.critical)
        }

        button("Danger!") {
            addClass(MyStyle.critical)
        }
    }
}
图6.8

现在你在HBox周围有橙色边框以及按钮。 当嵌套样式时,请记住,使用and()包装选择将级联样式到子级控件或类(cascade styles to children controls or classes)。

混入(Mixins)

有时您可能想要重复使用一组样式,并将它们应用于多个控件和选择器。 这样就不必冗余地定义相同的属性和值。 例如,如果要创建一组称为redAllTheThings的样式,可以将其定义为mixin,如下所示。 然后,您可以将其重新使用为redStyle类,以及一个textInput,一个label和一个具有附加样式修改的passwordField (图6.9)。

样式表

import javafx.scene.paint.Color
import javafx.scene.text.FontWeight
import tornadofx.*

class Styles : Stylesheet() {

    companion object {
        val redStyle by cssclass().
    }

    init {
        val redAllTheThings = mixin {
            backgroundInsets += box(5.px)
            borderColor += box(Color.RED)
            textFill = Color.RED
        }

        redStyle {
            +redAllTheThings
        }

        s(textInput, label) {
            +redAllTheThings
            fontWeight = FontWeight.BOLD
        }

        passwordField {
            +redAllTheThings
            backgroundColor += Color.YELLOW
        }
    }
}

应用和视图:

class MyApp: App(MyView::class, Styles::class)

class MyView : View("My View") {
    override val root = vbox {
        label("Enter your login")
        form {
            fieldset{
                field("Username") {
                    textfield()
                }
                field("Password") {
                    passwordfield()
                }
            }
        }
        button("Go!") {
            addClass(Styles.redStyle)
        }
    }
}
图6.9

样式表通过将其作为构造函数参数添加到App类应用于应用程序。 这是一个·vararg·参数,因此您可以以逗号分隔的多个样式表列表发送。 如果要根据某些条件动态加载样式表,可以从任何地方调用importStylesheet(Styles::class) 。在调用importStylesheet之后打开的任何UIComponent importStylesheet将获取应用的样式表,还可以使用此功能加载基于正常文本的css样式表:

importStylesheet("/mystyles.css")

加载基于文本的CSS样式表

如果您发现自己将相同的CSS属性重新设置为相同的值,则可能需要考虑使用 mixins并在Stylesheet重用它们。

修饰符选择(Modifier Selections)

TornadoFX还通过在选择中利用and()函数来支持修饰符选择。 最常见的情况是方便用于“选择”("selected")和游标“悬停”("hover")上下文的样式。

如果你想创建一个UI,当它被悬停在其上时,任何一个Button上,并且数据控件(如ListView所有选择的Cell都可以定义一个Stylesheet如图6.10所示。

样式表


import javafx.scene.paint.Color
import tornadofx.Stylesheet

class Styles : Stylesheet() {

    init {
        button {
            and(hover) {
                backgroundColor += Color.RED
            }
        }
        cell {
            and(selected) {
                backgroundColor += Color.RED
            }
        }
    }
}

应用和视图

import tornadofx.*

class MyApp: App(MyView::class, Styles::class)

class MyView : View("My View") {

    val listItems = listOf("Alpha","Beta","Gamma").observable()
and
    override val root = vbox {
        button("Hover over me")
        listview(listItems)
    }
}

图6.10 - 选择一个单元格,并且该Button被悬停在上面。 现在都是红色的。

图6.10

无论何时需要修饰符,请使用select()函数来进行上下文风格的修改。

控件特定样式表(Control-Specific Stylesheets)

如果您决定创建自己的控件(通常通过扩展现有控件,如Button ),JavaFX允许您将样式表与它进行配对。 在这种情况下,仅当加载此控件时,加载此Stylesheet是有利的。 例如,如果您有一个DangerButton类扩展Button ,您可以考虑为该DangerButton专门创建Stylesheet 。 要允许JavaFX加载它,您需要覆盖getUserAgentStyleSheet()函数,如下所示。 这将将您的类型安全的Stylesheet转换为JavaFX本身所理解的纯文本CSS。

class DangerButton : Button("Danger!") {
    init {
        addClass(DangerButtonStyles.dangerButton)
    }
    override fun getUserAgentStylesheet() = DangerButtonStyles().base64URL.toExternalForm()
}

class DangerButtonStyles : Stylesheet() {
    companion object {
        val dangerButton by cssclass()
    }

    init {
        dangerButton {
            backgroundInsets += box(0.px)
            fontWeight = FontWeight.BOLD
            fontSize = 20.px
            padding = box(10.px)
        }
    }
}

DangerButtonStyles().base64URL.toExternalForm()表达式创建一个DangerButtonStyles的实例,并将其转换为包含JavaFX可以使用的整个样式表的URL。

总结

TornadoFX做了一个伟大的工作,执行一个聪明的概念,使CSS类型安全,并进一步展示了Kotlin DSL的力量。 通过静态文本文件进行配置是很慢的,但类型安全的CSS使得其流畅且快速,特别是使用IDE自动完成。 即使你对UI是务实的,感觉样式是多余的,有时候你需要利用条件格式化和突出显示,以便在UI中弹出规则。 至少可以使用内嵌style { }块,以便您可以快速访问无法以其他方式访问的样式属性(例如TextWeight )。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容