Scala 使用sbt native packager和JDKPackager插件定制Freedesktop文件

Scala 使用sbt native packager和JDKPackager插件定制Freedesktop文件,scala,sbt-native-packager,Scala,Sbt Native Packager,我想定制由javapackager创建的.desktop文件,作为sbt native packager的JDKPackager插件的一部分。它显然使用了一个模板: [info]使用默认包资源[菜单快捷方式描述符] (将package/linux/Foo.desktop添加到类路径以进行自定义) 特别是,我想添加StartupWMClass条目,Gnome将使用该条目来统一我的应用程序打开的所有窗口。javapackager引用插件的目标目录,即target/jdkpackager。例如,当编

我想定制由javapackager创建的
.desktop
文件,作为sbt native packager的JDKPackager插件的一部分。它显然使用了一个模板:

[info]使用默认包资源[菜单快捷方式描述符]
(将package/linux/Foo.desktop添加到类路径以进行自定义)

特别是,我想添加
StartupWMClass
条目,Gnome将使用该条目来统一我的应用程序打开的所有窗口。

javapackager引用插件的目标目录,即
target/jdkpackager
。例如,当编写javafx ant构建文件时,就会创建该文件。因此,我们可以在这里进行:

// rewrite the task so that after the ant build is created,
// we add package/linux/MyApp.desktop
writeAntBuild in JDKPackager := {
  val res  = (writeAntBuild in JDKPackager).value
  val main = (mainClass     in JDKPackager).value
    .getOrElse(sys.error("No main class specified"))
  val tgt  = (target        in JDKPackager).value
  val n    = (name          in JDKPackager).value
  val wm   = main.replace('.', '-')
  val desktop = 
    s"""[Desktop Entry]
       |Name=APPLICATION_NAME
       |Comment=APPLICATION_SUMMARY
       |Exec=/opt/APPLICATION_FS_NAME/APPLICATION_LAUNCHER_FILENAME
       |Icon=/opt/APPLICATION_FS_NAME/APPLICATION_LAUNCHER_FILENAME.png
       |Terminal=false
       |Type=Application
       |Categories=DEPLOY_BUNDLE_CATEGORY
       |DESKTOP_MIMES
       |StartupWMClass=$wm
       |""".stripMargin
  IO.write(tgt / "package" / "linux" / s"$n.desktop", desktop)
  res
}

只要在中有相应的设置,就有另一种解决方案:只需通过
antBuildDefn
重写插件生成的XML即可

下面是通过添加以下内容为
.desktop
文件指定菜单类别的示例:


从Oracle
javapackager
文档中并不完全清楚这一点,原因是
build.xml
文件将
tgt
目录放在Ant任务类路径中,然后在其类路径的
package/linux
分支中查找覆盖资源。
antBuildDefn in JDKPackager := {
  val origTask = (antBuildDefn in JDKPackager).value

  val InfoLabel = "info"
  val KeyRegex  = s"$InfoLabel\\.(.+)".r

  import scala.xml._
  import scala.xml.transform._
  val infoRewrite = new RewriteRule {
    override def transform(n: Node) = n match {
      case e: Elem if e.prefix == "fx" && e.label == InfoLabel =>
        e % Attribute("", "category", "Office", Null)
      case other => other
    }
  }

  new RuleTransformer(infoRewrite)(origTask)
}