TornadoFX編程指南,第12章,OSGi

譯自《OSGi

OSGi

本章主要面向已經(jīng)熟悉OSGi的人 ,代表開放式服務(wù)網(wǎng)關(guān)計劃(Open Services Gateway Initiative) 。 OSGi背后的想法是將模塊添加到Java應(yīng)用程序中,而不需要重新啟動。 TornadoFX支持OSGi,并允許高度模塊化和動態(tài)的應(yīng)用。

如果您對OSGi目前沒有興趣,歡迎跳過本章。 然而,強烈建議至少知道它是什么,這樣您就可以識別將來可能需要使用的時刻。

TornadoFX帶有OSGi運行時所需的元數(shù)據(jù),以檢測和啟用它。 當(dāng)tornadofx.jar加載到OSGi容器中時,會在運行時自動安裝多個服務(wù)。 這些服務(wù)使我們能夠討論一些非常有趣的功能。

OSGi 簡介

在繼續(xù)本章之前,請先熟悉OSGi的基礎(chǔ)知識。 要快速了解OSGi技術(shù),您可以查看OSGi聯(lián)盟網(wǎng)站(OSGi Alliance website)上的教程 (tutorials)。 Apache Felix教程(Apache Felix tutorials)也是基本OSGi模式的良好參考起點。

服務(wù)(Services)

當(dāng)TornadoFX JAR加載時,您可以創(chuàng)建自己的TornadoFX軟件包(bundle),并以任何您喜歡的方式創(chuàng)建應(yīng)用程序。 然而,一些使用模式是非常典型和有用的,因此TornadoFX具有內(nèi)置的對他們的支持。

動態(tài)應(yīng)用(Dynamic Applications)

OSGi的動態(tài)性質(zhì)一般適用于GUI應(yīng)用程序。 隨著環(huán)境的變化,具有某些功能的加入或撤出( come and go)的能力可以很強大。 不幸的是,JavaFX本身是以一種防止在初始的應(yīng)用程序關(guān)閉后啟動另一個JavaFX應(yīng)用程序的方式編寫的。 為了規(guī)避這個缺點,您可以根據(jù)需要停止和啟動應(yīng)用程序,TornadoFX提供了一種使用應(yīng)用程序代理(application proxy)注冊App類的方法,即使在您的應(yīng)用程序關(guān)閉時,也能保持JavaFX環(huán)境運行。

要開始,可以實現(xiàn)一個BundleActivator,它提供start()stop() 一個App的方法 。 您的bundle的 Activator中,可以通過調(diào)用context.registerApplication來注冊您的應(yīng)用程序的這個功能,并使用您的App類作為單個參數(shù)完成:

class Activator : BundleActivator {
    override fun start(context: BundleContext) {
        context.registerApplication(MyApp::class)
    }

    override fun stop(context: BundleContext) {
    }
}

如果您更喜歡使用OSGi聲明性服務(wù)(declarative services),下面的方式會具有相同的效果,只要您加載了OSGi DS軟件包:

@Component
class AppRegistration : ApplicationProvider {
    override val application = MyApp::class
}

如果您的容器中提供了TornadoFX軟件包,這足以在軟件包激活后自動啟動應(yīng)用程序。 您現(xiàn)在可以通過停止或啟動軟件包(bundle)來停止并啟動多次你的應(yīng)用。

動態(tài)樣式表

要為其他TornadoFX軟件包提供類型安全的樣式表(type-safe stylesheets),您可以通過在Activator中注冊他們:

class Activator : BundleActivator {
    override fun start(context: BundleContext) {
        context.registerStylesheet(Styles::class)
    }

    override fun stop(context: BundleContext) {
    }
}

使用OSGi聲明式服務(wù)(Declarative Services),注冊如下所示:

@Component
class StyleRegistration : StylesheetProvider {
    override val stylesheet = Styles::class
}

每當(dāng)加載此bundle時,每個活動View都將應(yīng)用此樣式表。 該bundle卸載后,樣式表也將自動刪除。 如果要根據(jù)相同的樣式類(style classes)提供多個樣式表(multiple stylesheets),最好創(chuàng)建一個導(dǎo)出cssclass定義的bundle,以便您的Views可以引用這些樣式,并且樣式表包(stylesheet bundles)可以根據(jù)它們創(chuàng)建選擇器(selectors)。

動態(tài)視圖

OSGi的一個很酷的方面是在UI元素可用時就彈出UI元素的能力。 典型的用例是“儀表板(dashboard)”應(yīng)用程序。 在此示例中,基礎(chǔ)應(yīng)用程序包(base application bundle)包含可以保存其他視圖(Views)的View,并告訴TornadoFX OSGi運行時(Runtime),如果符合特定條件,它將自動嵌入視圖(Views)。

例如,我們可以創(chuàng)建一個包含VBoxView。 我們告訴TornadoFX OSGi運行時,我們希望將其他視圖(Views)嵌入其中,如果它們使用了discriminator儀表板(dashboard)標(biāo)記:

class Dashboard : View() {
    override val root = VBox()

    init {
        title = "Dashboard Application"
        addViewsWhen { it.discriminator == "dashboard" }
    }
}

如果addViewsWhen函數(shù)返回true,則將View添加到VBox 。 要提供對此儀表板(Dashboard)的視圖(Views),另一個包(bundle)可以聲明它要通過設(shè)置dashboarddiscriminator導(dǎo)出它的視圖。 在這里,我們注冊一個虛構(gòu)的MusicPlayer視圖,當(dāng)它的包(bundle)變?yōu)榛顒訒r,就會被放置在儀表板中( docked into the dashboard)。

class Activator : BundleActivator {
    override fun start(context: BundleContext) {
        context.registerView(MusicPlayer::class, "dashboard")
    }

    override fun stop(context: BundleContext) {
    }
}

再次,導(dǎo)出視圖的OSGi Declarative Services方式如下所示:

@Component
class MusicPlayerRegistration : ViewProvider {
    override val discriminator = "dashboard"
    override fun getView() = find(MusicPlayer::class)
}

addViewsWhen函數(shù)足夠聰明以檢查VBox并找出如何添加子視圖。 還可以看出,如果您在TabPane上調(diào)用該函數(shù),它將創(chuàng)建一個新的Tab并將標(biāo)題設(shè)置為子視圖的標(biāo)題等。如果您想使用所提供的視圖進(jìn)行自定義,則可以從函數(shù)返回false,以便不會自動添加子視圖,然后就可以執(zhí)行所需的任何操作。 即使Tab示例是開箱即用的,您也可以明確地這樣做:

tabPane.addViewsWhen {
    if (it.discriminator == "dashboard") {
        val view = it.getView()
        tabPane.tab(view.title, view.root)
    }
    false
}

手動處理動態(tài)視圖

創(chuàng)建您的第一個OSGi包

一個好的起點是TornadoFX IntelliJ IDEA插件中的tornadofx-maven-osgi-project模板。 這包含從您的源代碼來構(gòu)建OSGi軟件包所需的一切。 OSGI IDEA插件使得從IDE直接設(shè)置和運行OSGi容器非常容易。 https://www.youtube.com/watch?v=liOFCH5MMKK上有一個屏幕截圖,顯示了這些概念的運用。

OSGi控制臺

TornadoFX有一個內(nèi)置的OSGi控制臺,您可以從中檢查軟件包,更改其狀態(tài),甚至通過拖放來安裝新的軟件包。 您可以使用Alt-Meta-O打開控制臺,或通過設(shè)置FX.osgiConsoleShortcut或以編程方式打開OSGIConsole View來配置其他快捷方式。

要求

要在OSGi容器中運行TornadoFX,需要加載所需的包(bundles)。 通常只是需要將這些jar轉(zhuǎn)儲到容器的bundle目錄中就行了。 請注意,要在OSGi容器中使用的任何jar需要“啟用OSGi”(OSGi enabled),這實際上意味著添加一些OSGi特定條目到META-INF/MANIFEST.MF文件。

我們提供了一個完整的安裝,其中Apache Felix和TornadoFX已經(jīng)安裝在http://tornadofx.tornado.no/felix-tornadofx-5.4.0.zip 。 記得切換tornadofx.jar的最新版本,因為這個bundle最有可能落后于幾個版本。

這些是在OSGi容器中運行任何TornadoFX應(yīng)用程序所需的工件。 您的容器可能已經(jīng)與其中的一些捆綁在一起,因此請檢查容器文檔以獲取更多詳細(xì)信息。

    1. Kotlin OSGi bundle 包含特定版本的kotlin-stdlibkotlin-reflect,具有所需的OSGi清單(manifest)信息。
    1. 這鏈接到Apache Felix實現(xiàn)的OSGi配置管理界面(OSGi Config Admin interface)。 隨意使用你的OSGi容器的實現(xiàn)。 某些容器(如Apache Karaf)已經(jīng)裝載了Config Admin bundle,因此您不需要它。
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 134,937評論 18 139
  • 場景 單機應(yīng)用已經(jīng)越來越不能符合目前越來越復(fù)雜的產(chǎn)品需求了。即使是小型應(yīng)用,至少也需要部署2臺以上的服務(wù)器做集群。...
    IvanEye閱讀 3,625評論 3 11
  • 關(guān)于iOS應(yīng)用程序架構(gòu) 應(yīng)用程序需要與iOS一起工作,以確保他們提供出色的用戶體驗。 除了為您的應(yīng)用程序的設(shè)計和用...
    nicedayCoco閱讀 1,243評論 0 1
  • 在您的應(yīng)用程序中啟用狀態(tài)保存和恢復(fù) 狀態(tài)保存和恢復(fù)不是自動功能,應(yīng)用程序必須選擇使用它。應(yīng)用程序通過在其應(yīng)用程序委...
    nicedayCoco閱讀 528評論 0 0
  • 關(guān)于首選項和設(shè)置 首選項是您持久存儲的信息,并用于配置您的應(yīng)用程序。應(yīng)用程序通常會向用戶公開偏好設(shè)置,以便他們自定...
    nicedayCoco閱讀 1,031評論 0 0