diff --git a/src/jdk.jpackage/share/classes/jdk/jpackage/internal/Arguments.java b/src/jdk.jpackage/share/classes/jdk/jpackage/internal/Arguments.java index 574d49cef37..b51e6e61f04 100644 --- a/src/jdk.jpackage/share/classes/jdk/jpackage/internal/Arguments.java +++ b/src/jdk.jpackage/share/classes/jdk/jpackage/internal/Arguments.java @@ -330,6 +330,11 @@ public class Arguments { setOptionValue("win-shortcut", true); }), + WIN_SHORTCUT_PROMPT ("win-shortcut-prompt", + OptionCategories.PLATFORM_WIN, () -> { + setOptionValue("win-shortcut-prompt", true); + }), + WIN_PER_USER_INSTALLATION ("win-per-user-install", OptionCategories.PLATFORM_WIN, () -> { setOptionValue("win-per-user-install", false); diff --git a/src/jdk.jpackage/share/classes/jdk/jpackage/internal/ValidOptions.java b/src/jdk.jpackage/share/classes/jdk/jpackage/internal/ValidOptions.java index 5fb8a68fbe2..0a12f6cb7db 100644 --- a/src/jdk.jpackage/share/classes/jdk/jpackage/internal/ValidOptions.java +++ b/src/jdk.jpackage/share/classes/jdk/jpackage/internal/ValidOptions.java @@ -96,6 +96,7 @@ class ValidOptions { options.put(CLIOptions.WIN_MENU_HINT.getId(), USE.INSTALL); options.put(CLIOptions.WIN_MENU_GROUP.getId(), USE.INSTALL); options.put(CLIOptions.WIN_SHORTCUT_HINT.getId(), USE.INSTALL); + options.put(CLIOptions.WIN_SHORTCUT_PROMPT.getId(), USE.INSTALL); options.put(CLIOptions.WIN_DIR_CHOOSER.getId(), USE.INSTALL); options.put(CLIOptions.WIN_UPGRADE_UUID.getId(), USE.INSTALL); options.put(CLIOptions.WIN_PER_USER_INSTALLATION.getId(), diff --git a/src/jdk.jpackage/share/classes/jdk/jpackage/internal/resources/HelpResources.properties b/src/jdk.jpackage/share/classes/jdk/jpackage/internal/resources/HelpResources.properties index baabcfe526d..f793835f27c 100644 --- a/src/jdk.jpackage/share/classes/jdk/jpackage/internal/resources/HelpResources.properties +++ b/src/jdk.jpackage/share/classes/jdk/jpackage/internal/resources/HelpResources.properties @@ -1,5 +1,5 @@ # -# Copyright (c) 2017, 2020, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2017, 2021, Oracle and/or its affiliates. All rights reserved. # DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. # # This code is free software; you can redistribute it and/or modify it @@ -205,13 +205,16 @@ MSG_Help_win_install=\ \ Adds a dialog to enable the user to choose a directory in which\n\ \ the product is installed\n\ \ --win-menu\n\ -\ Adds the application to the system menu\n\ +\ Request to add a Start menu shortcut for this application\n\ \ --win-menu-group \n\ \ Start Menu group this application is placed in\n\ \ --win-per-user-install\n\ \ Request to perform an install on a per-user basis\n\ \ --win-shortcut\n\ -\ Creates a desktop shortcut for the application\n\ +\ Request to add desktop shortcut for this application\n\ +\ --win-shortcut-prompt\n\ +\ Adds a dialog to enable the user to choose if shortcuts\n\ +\ will be created by installer\n\ \ --win-upgrade-uuid \n\ \ UUID associated with upgrades for this package\n\ diff --git a/src/jdk.jpackage/share/classes/jdk/jpackage/internal/resources/HelpResources_ja.properties b/src/jdk.jpackage/share/classes/jdk/jpackage/internal/resources/HelpResources_ja.properties index aadbb79d215..1b13545eda0 100644 --- a/src/jdk.jpackage/share/classes/jdk/jpackage/internal/resources/HelpResources_ja.properties +++ b/src/jdk.jpackage/share/classes/jdk/jpackage/internal/resources/HelpResources_ja.properties @@ -31,7 +31,24 @@ MSG_Help=\u4F7F\u7528\u65B9\u6CD5: jpackage \n\n\u4F7F\u7528\u4F8B:\n-- \u3053\u306E\u30AA\u30D7\u30B7\u30E7\u30F3\u306F\u8907\u6570\u56DE\u4F7F\u7528\u3067\u304D\u307E\u3059\u3002\n --install-dir \n {4} --license-file \n \u30E9\u30A4\u30BB\u30F3\u30B9\u30FB\u30D5\u30A1\u30A4\u30EB\u3078\u306E\u30D1\u30B9\n (\u7D76\u5BFE\u30D1\u30B9\u307E\u305F\u306F\u73FE\u5728\u306E\u30C7\u30A3\u30EC\u30AF\u30C8\u30EA\u304B\u3089\u306E\u76F8\u5BFE\u30D1\u30B9)\n --resource-dir \n \u30AA\u30FC\u30D0\u30FC\u30E9\u30A4\u30C9jpackage\u30EA\u30BD\u30FC\u30B9\u3078\u306E\u30D1\u30B9\n \u30A2\u30A4\u30B3\u30F3\u3001\u30C6\u30F3\u30D7\u30EC\u30FC\u30C8\u30FB\u30D5\u30A1\u30A4\u30EB\u304A\u3088\u3073jpackage\u306E\u305D\u306E\u4ED6\u306E\u30EA\u30BD\u30FC\u30B9\u306F\u3001\n \u3053\u306E\u30C7\u30A3\u30EC\u30AF\u30C8\u30EA\u306B\u7F6E\u63DB\u30EA\u30BD\u30FC\u30B9\u3092\u8FFD\u52A0\u3059\u308B\u3053\u3068\u3067\u30AA\u30FC\u30D0\u30FC\u30E9\u30A4\u30C9\u3067\u304D\u307E\u3059\u3002\n (\u7D76\u5BFE\u30D1\u30B9\u307E\u305F\u306F\u73FE\u5728\u306E\u30C7\u30A3\u30EC\u30AF\u30C8\u30EA\u304B\u3089\u306E\u76F8\u5BFE\u30D1\u30B9)\n --runtime-image \n \u30A4\u30F3\u30B9\u30C8\u30FC\u30EB\u3059\u308B\u4E8B\u524D\u5B9A\u7FA9\u6E08\u307F\u306E\u30E9\u30F3\u30BF\u30A4\u30E0\u30FB\u30A4\u30E1\u30FC\u30B8\u306E\u30D1\u30B9\n (\u7D76\u5BFE\u30D1\u30B9\u307E\u305F\u306F\u73FE\u5728\u306E\u30C7\u30A3\u30EC\u30AF\u30C8\u30EA\u304B\u3089\u306E\u76F8\u5BFE\u30D1\u30B9)\n \u30E9\u30F3\u30BF\u30A4\u30E0\u30FB\u30D1\u30C3\u30B1\u30FC\u30B8\u306E\u4F5C\u6210\u6642\u306B\u306F\u3001\u30AA\u30D7\u30B7\u30E7\u30F3\u304C\u5FC5\u8981\u3067\u3059\u3002\n\n\u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u30FB\u30D1\u30C3\u30B1\u30FC\u30B8\u3092\u4F5C\u6210\u3059\u308B\u305F\u3081\u306E\u30D7\u30E9\u30C3\u30C8\u30D5\u30A9\u30FC\u30E0\u4F9D\u5B58\u30AA\u30D7\u30B7\u30E7\u30F3:\n{3} MSG_Help_win_launcher=\n\u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u30FB\u30E9\u30F3\u30C1\u30E3\u3092\u4F5C\u6210\u3059\u308B\u305F\u3081\u306E\u30D7\u30E9\u30C3\u30C8\u30D5\u30A9\u30FC\u30E0\u4F9D\u5B58\u30AA\u30D7\u30B7\u30E7\u30F3:\n --win-console\n \u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u306E\u30B3\u30F3\u30BD\u30FC\u30EB\u30FB\u30E9\u30F3\u30C1\u30E3\u3092\u4F5C\u6210\u3057\u307E\u3059\u3002\u30B3\u30F3\u30BD\u30FC\u30EB\u30FB\n \u30A4\u30F3\u30BF\u30E9\u30AF\u30B7\u30E7\u30F3\u304C\u5FC5\u8981\u306A\u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u306B\u6307\u5B9A\u3059\u308B\u5FC5\u8981\u304C\u3042\u308A\u307E\u3059\n -MSG_Help_win_install=\ --win-dir-chooser\n \u30E6\u30FC\u30B6\u30FC\u304C\u88FD\u54C1\u3092\u30A4\u30F3\u30B9\u30C8\u30FC\u30EB\u3059\u308B\u30C7\u30A3\u30EC\u30AF\u30C8\u30EA\u3092\u9078\u629E\n \u3059\u308B\u305F\u3081\u306E\u30C0\u30A4\u30A2\u30ED\u30B0\u3092\u8FFD\u52A0\u3057\u307E\u3059\n --win-menu\n \u30B7\u30B9\u30C6\u30E0\u30FB\u30E1\u30CB\u30E5\u30FC\u306B\u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u3092\u8FFD\u52A0\u3057\u307E\u3059\n --win-menu-group \n \u3053\u306E\u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u3092\u914D\u7F6E\u3059\u308B\u30B9\u30BF\u30FC\u30C8\u30FB\u30E1\u30CB\u30E5\u30FC\u30FB\u30B0\u30EB\u30FC\u30D7\n --win-per-user-install\n \u30E6\u30FC\u30B6\u30FC\u3054\u3068\u306B\u30A4\u30F3\u30B9\u30C8\u30FC\u30EB\u3092\u5B9F\u884C\u3059\u308B\u3088\u3046\u306B\u30EA\u30AF\u30A8\u30B9\u30C8\u3057\u307E\u3059\n --win-shortcut\n \u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u306E\u30C7\u30B9\u30AF\u30C8\u30C3\u30D7\u30FB\u30B7\u30E7\u30FC\u30C8\u30AB\u30C3\u30C8\u3092\u4F5C\u6210\u3057\u307E\u3059\n --win-upgrade-uuid \n \u3053\u306E\u30D1\u30C3\u30B1\u30FC\u30B8\u306E\u30A2\u30C3\u30D7\u30B0\u30EC\u30FC\u30C9\u306B\u95A2\u9023\u4ED8\u3051\u3089\u308C\u3066\u3044\u308BUUID\n +MSG_Help_win_install=\ +\ --win-dir-chooser\n\ +\ Adds a dialog to enable the user to choose a directory in which\n\ +\ the product is installed\n\ +\ --win-menu\n\ +\ Request to add a Start menu shortcut for this application\n\ +\ --win-menu-group \n\ +\ Start Menu group this application is placed in\n\ +\ --win-per-user-install\n\ +\ Request to perform an install on a per-user basis\n\ +\ --win-shortcut\n\ +\ Request to add desktop shortcut for this application\n\ +\ --win-shortcut-prompt\n\ +\ Adds a dialog to enable the user to choose if shortcuts\n\ +\ will be created by installer\n\ +\ --win-upgrade-uuid \n\ +\ UUID associated with upgrades for this package\n\ + MSG_Help_win_install_dir=\u30C7\u30D5\u30A9\u30EB\u30C8\u306E\u30A4\u30F3\u30B9\u30C8\u30FC\u30EB\u5834\u6240\u306E\u4E0B\u306E\u76F8\u5BFE\u30B5\u30D6\u30D1\u30B9\n MSG_Help_mac_launcher=\ --mac-package-identifier \n MacOS\u306E\u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u3092\u4E00\u610F\u306B\u8B58\u5225\u3059\u308BID\n \u30E1\u30A4\u30F3\u30FB\u30AF\u30E9\u30B9\u540D\u306B\u30C7\u30D5\u30A9\u30EB\u30C8\u8A2D\u5B9A\u3055\u308C\u3066\u3044\u307E\u3059\u3002\n \u82F1\u6570\u5B57(A-Z\u3001a-z\u30010-9)\u3001\u30CF\u30A4\u30D5\u30F3(-)\n \u304A\u3088\u3073\u30D4\u30EA\u30AA\u30C9(.)\u6587\u5B57\u306E\u307F\u4F7F\u7528\u3067\u304D\u307E\u3059\u3002\n --mac-package-name \n \u30E1\u30CB\u30E5\u30FC\u30FB\u30D0\u30FC\u306B\u8868\u793A\u3055\u308C\u308B\u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u306E\u540D\u524D\n \u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u540D\u3068\u306F\u7570\u306A\u308A\u307E\u3059\u3002\n \u3053\u306E\u540D\u524D\u306F16\u6587\u5B57\u672A\u6E80\u306B\u3059\u308B\u5FC5\u8981\u304C\u3042\u308A\u3001\u30E1\u30CB\u30E5\u30FC\u30FB\u30D0\u30FC\n \u304A\u3088\u3073\u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u60C5\u5831\u30A6\u30A3\u30F3\u30C9\u30A6\u306B\u8868\u793A\u3059\u308B\u306E\u306B\u9069\u3057\u3066\u3044\u308B\u5FC5\u8981\u304C\u3042\u308A\u307E\u3059\u3002\n \u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u540D\u306B\u30C7\u30D5\u30A9\u30EB\u30C8\u8A2D\u5B9A\u3055\u308C\u3066\u3044\u307E\u3059\u3002\n --mac-package-signing-prefix \n \u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u30FB\u30D1\u30C3\u30B1\u30FC\u30B8\u306B\u7F72\u540D\u3059\u308B\u969B\u3001\u65E2\u5B58\u306E\u30D1\u30C3\u30B1\u30FC\u30B8ID\u306E\u306A\u3044\n \u7F72\u540D\u304C\u5FC5\u8981\u306A\u3059\u3079\u3066\u306E\u30B3\u30F3\u30DD\u30FC\u30CD\u30F3\u30C8\u306B\u3001\n \u3053\u306E\u5024\u304C\u63A5\u982D\u8F9E\u3068\u3057\u3066\u4ED8\u3051\u3089\u308C\u307E\u3059\u3002\n --mac-sign\n \u30D1\u30C3\u30B1\u30FC\u30B8\u306B\u7F72\u540D\u3059\u308B\u3088\u3046\u30EA\u30AF\u30A8\u30B9\u30C8\u3057\u307E\u3059\n --mac-signing-keychain \n \u7F72\u540D\u30A2\u30A4\u30C7\u30F3\u30C6\u30A3\u30C6\u30A3\u3092\u691C\u7D22\u3059\u308B\u30AD\u30FC\u30C1\u30A7\u30FC\u30F3\u306E\u30D1\u30B9\n (\u7D76\u5BFE\u30D1\u30B9\u307E\u305F\u306F\u73FE\u5728\u306E\u30C7\u30A3\u30EC\u30AF\u30C8\u30EA\u304B\u3089\u306E\u76F8\u5BFE\u30D1\u30B9)\u3002\n \u6307\u5B9A\u3057\u306A\u304B\u3063\u305F\u5834\u5408\u3001\u6A19\u6E96\u306E\u30AD\u30FC\u30C1\u30A7\u30FC\u30F3\u304C\u4F7F\u7528\u3055\u308C\u307E\u3059\u3002\n --mac-signing-key-user-name \n Apple\u7F72\u540D\u30A2\u30A4\u30C7\u30F3\u30C6\u30A3\u30C6\u30A3\u306E\u540D\u524D\u306E\u30C1\u30FC\u30E0\u540D\u90E8\u5206\u3002\n \u4F8B: "Developer ID Application: "\n MSG_Help_linux_install=\ --linux-package-name \n Linux\u30D1\u30C3\u30B1\u30FC\u30B8\u306E\u540D\u524D\u3002\u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u540D\u306B\u30C7\u30D5\u30A9\u30EB\u30C8\u8A2D\u5B9A\u3055\u308C\u3066\u3044\u307E\u3059\n --linux-deb-maintainer \n .deb\u30D1\u30C3\u30B1\u30FC\u30B8\u306EMaintainer\n --linux-menu-group \n \u3053\u306E\u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u304C\u914D\u7F6E\u3055\u308C\u3066\u3044\u308B\u30E1\u30CB\u30E5\u30FC\u30FB\u30B0\u30EB\u30FC\u30D7\n --linux-package-deps\n \u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u306B\u5FC5\u8981\u306A\u30D1\u30C3\u30B1\u30FC\u30B8\u307E\u305F\u306F\u6A5F\u80FD\n --linux-rpm-license-type \n \u30E9\u30A4\u30BB\u30F3\u30B9\u306E\u30BF\u30A4\u30D7(RPM .spec\u306E"License: ")\n --linux-app-release \n RPM .spec\u30D5\u30A1\u30A4\u30EB\u306E\u30EA\u30EA\u30FC\u30B9\u5024\u307E\u305F\u306F \n DEB\u30B3\u30F3\u30C8\u30ED\u30FC\u30EB\u30FB\u30D5\u30A1\u30A4\u30EB\u306EDebian\u30EA\u30D3\u30B8\u30E7\u30F3\u5024\u3002\n --linux-app-category \n RPM .spec\u30D5\u30A1\u30A4\u30EB\u306E\u30B0\u30EB\u30FC\u30D7\u5024\u307E\u305F\u306F \n DEB\u30B3\u30F3\u30C8\u30ED\u30FC\u30EB\u30FB\u30D5\u30A1\u30A4\u30EB\u306E\u30BB\u30AF\u30B7\u30E7\u30F3\u5024\u3002\n --linux-shortcut\n \u30A2\u30D7\u30EA\u30B1\u30FC\u30B7\u30E7\u30F3\u306E\u30B7\u30E7\u30FC\u30C8\u30AB\u30C3\u30C8\u3092\u4F5C\u6210\u3057\u307E\u3059\n diff --git a/src/jdk.jpackage/share/classes/jdk/jpackage/internal/resources/HelpResources_zh_CN.properties b/src/jdk.jpackage/share/classes/jdk/jpackage/internal/resources/HelpResources_zh_CN.properties index bff1a8928f7..323611ab276 100644 --- a/src/jdk.jpackage/share/classes/jdk/jpackage/internal/resources/HelpResources_zh_CN.properties +++ b/src/jdk.jpackage/share/classes/jdk/jpackage/internal/resources/HelpResources_zh_CN.properties @@ -30,7 +30,24 @@ MSG_Help=\u7528\u6CD5\uFF1Ajpackage \n\n\u793A\u4F8B\u7528\u6CD5:\n---- \uFF08\u7EDD\u5BF9\u8DEF\u5F84\u6216\u76F8\u5BF9\u4E8E\u5F53\u524D\u76EE\u5F55\u7684\u8DEF\u5F84\uFF09\n \u5728\u521B\u5EFA\u8FD0\u884C\u65F6\u7A0B\u5E8F\u5305\u65F6\u9700\u8981\u4F7F\u7528\u9009\u9879\u3002\n\n\u7528\u6765\u521B\u5EFA\u5E94\u7528\u7A0B\u5E8F\u5305\u7684\u4E0E\u5E73\u53F0\u76F8\u5173\u7684\u9009\u9879\uFF1A\n{3} MSG_Help_win_launcher=\n\u7528\u6765\u521B\u5EFA\u5E94\u7528\u7A0B\u5E8F\u542F\u52A8\u7A0B\u5E8F\u7684\u4E0E\u5E73\u53F0\u76F8\u5173\u7684\u9009\u9879\uFF1A\n --win-console\n \u4E3A\u5E94\u7528\u7A0B\u5E8F\u521B\u5EFA\u63A7\u5236\u53F0\u542F\u52A8\u7A0B\u5E8F\uFF0C\u5E94\u5F53\u4E3A\n \u9700\u8981\u63A7\u5236\u53F0\u4EA4\u4E92\u7684\u5E94\u7528\u7A0B\u5E8F\u6307\u5B9A\n -MSG_Help_win_install=\ --win-dir-chooser\n \u6DFB\u52A0\u4E00\u4E2A\u5BF9\u8BDD\u6846\u4EE5\u5141\u8BB8\u7528\u6237\u9009\u62E9\n \u4EA7\u54C1\u7684\u5B89\u88C5\u76EE\u5F55\n --win-menu\n \u5C06\u8BE5\u5E94\u7528\u7A0B\u5E8F\u6DFB\u52A0\u5230\u7CFB\u7EDF\u83DC\u5355\u4E2D\n --win-menu-group \n \u542F\u52A8\u8BE5\u5E94\u7528\u7A0B\u5E8F\u6240\u5728\u7684\u83DC\u5355\u7EC4\n --win-per-user-install\n \u8BF7\u6C42\u57FA\u4E8E\u6BCF\u4E2A\u7528\u6237\u6267\u884C\u5B89\u88C5\n --win-shortcut\n \u4E3A\u5E94\u7528\u7A0B\u5E8F\u521B\u5EFA\u684C\u9762\u5FEB\u6377\u65B9\u5F0F\n --win-upgrade-uuid \n \u4E0E\u6B64\u7A0B\u5E8F\u5305\u5347\u7EA7\u76F8\u5173\u8054\u7684 UUID\n +MSG_Help_win_install=\ +\ --win-dir-chooser\n\ +\ Adds a dialog to enable the user to choose a directory in which\n\ +\ the product is installed\n\ +\ --win-menu\n\ +\ Request to add a Start menu shortcut for this application\n\ +\ --win-menu-group \n\ +\ Start Menu group this application is placed in\n\ +\ --win-per-user-install\n\ +\ Request to perform an install on a per-user basis\n\ +\ --win-shortcut\n\ +\ Request to add desktop shortcut for this application\n\ +\ --win-shortcut-prompt\n\ +\ Adds a dialog to enable the user to choose if shortcuts\n\ +\ will be created by installer\n\ +\ --win-upgrade-uuid \n\ +\ UUID associated with upgrades for this package\n\ + MSG_Help_win_install_dir=\u9ED8\u8BA4\u5B89\u88C5\u4F4D\u7F6E\u4E0B\u9762\u7684\u76F8\u5BF9\u5B50\u8DEF\u5F84\n MSG_Help_mac_launcher=\ --mac-package-identifier \n \u7528\u6765\u552F\u4E00\u5730\u6807\u8BC6 macOS \u5E94\u7528\u7A0B\u5E8F\u7684\u6807\u8BC6\u7B26\n \u9ED8\u8BA4\u4E3A\u4E3B\u7C7B\u540D\u79F0\u3002\n \u53EA\u80FD\u4F7F\u7528\u5B57\u6BCD\u6570\u5B57\uFF08A-Z\u3001a-z\u30010-9\uFF09\u3001\u8FDE\u5B57\u7B26 (-) \u548C\n \u53E5\u70B9 (.) \u5B57\u7B26\u3002\n --mac-package-name \n \u51FA\u73B0\u5728\u83DC\u5355\u680F\u4E2D\u7684\u5E94\u7528\u7A0B\u5E8F\u540D\u79F0\n \u8FD9\u53EF\u4EE5\u4E0E\u5E94\u7528\u7A0B\u5E8F\u540D\u79F0\u4E0D\u540C\u3002\n \u6B64\u540D\u79F0\u7684\u957F\u5EA6\u5FC5\u987B\u5C0F\u4E8E 16 \u4E2A\u5B57\u7B26\uFF0C\u9002\u5408\n \u663E\u793A\u5728\u83DC\u5355\u680F\u4E2D\u548C\u5E94\u7528\u7A0B\u5E8F\u201C\u4FE1\u606F\u201D\u7A97\u53E3\u4E2D\u3002\n \u9ED8\u8BA4\u4E3A\u5E94\u7528\u7A0B\u5E8F\u540D\u79F0\u3002\n --mac-package-signing-prefix \n \u5728\u5BF9\u5E94\u7528\u7A0B\u5E8F\u5305\u7B7E\u540D\u65F6\uFF0C\u4F1A\u5728\u6240\u6709\u9700\u8981\u7B7E\u540D\n \u4F46\u5F53\u524D\u6CA1\u6709\u7A0B\u5E8F\u5305\u6807\u8BC6\u7B26\u7684\u7EC4\u4EF6\u7684\n \u524D\u9762\u52A0\u4E0A\u6B64\u503C\u3002\n --mac-sign\n \u8BF7\u6C42\u5BF9\u7A0B\u5E8F\u5305\u8FDB\u884C\u7B7E\u540D\n --mac-signing-keychain \n \u8981\u7528\u6765\u641C\u7D22\u7B7E\u540D\u8EAB\u4EFD\u7684\u5BC6\u94A5\u94FE\u7684\u8DEF\u5F84\n \uFF08\u7EDD\u5BF9\u8DEF\u5F84\u6216\u76F8\u5BF9\u4E8E\u5F53\u524D\u76EE\u5F55\u7684\u8DEF\u5F84\uFF09\u3002\n \u5982\u679C\u672A\u6307\u5B9A\uFF0C\u5219\u4F7F\u7528\u6807\u51C6\u7684\u5BC6\u94A5\u94FE\u3002\n --mac-signing-key-user-name \n Apple \u7B7E\u540D\u8EAB\u4EFD\u540D\u79F0\u4E2D\u7684\u56E2\u961F\u540D\u79F0\u9009\u9879\u3002\n \u4F8B\u5982\uFF0C"Developer ID Application: "\n MSG_Help_linux_install=\ --linux-package-name \n Linux \u7A0B\u5E8F\u5305\u7684\u540D\u79F0\uFF0C\u9ED8\u8BA4\u4E3A\u5E94\u7528\u7A0B\u5E8F\u540D\u79F0\n --linux-deb-maintainer \n .deb \u7A0B\u5E8F\u5305\u7684\u7EF4\u62A4\u7A0B\u5E8F\n --linux-menu-group \n \u6B64\u5E94\u7528\u7A0B\u5E8F\u6240\u5728\u7684\u83DC\u5355\u7EC4\n --linux-package-deps\n \u5E94\u7528\u7A0B\u5E8F\u6240\u9700\u7684\u7A0B\u5E8F\u5305\u6216\u529F\u80FD\n --linux-rpm-license-type \n \u8BB8\u53EF\u8BC1\u7684\u7C7B\u578B\uFF08RPM .spec \u7684 "License: "\uFF09\n --linux-app-release \n RPM .spec \u6587\u4EF6\u7684\u53D1\u884C\u7248\u503C\u6216\n DEB \u63A7\u5236\u6587\u4EF6\u7684 Debian \u4FEE\u8BA2\u7248\u503C\u3002\n --linux-app-category \n RPM .spec \u6587\u4EF6\u7684\u7EC4\u503C\u6216\n DEB \u63A7\u5236\u6587\u4EF6\u7684\u533A\u57DF\u503C\u3002\n --linux-shortcut\n \u4E3A\u5E94\u7528\u7A0B\u5E8F\u521B\u5EFA\u5FEB\u6377\u65B9\u5F0F\n diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WinMsiBundler.java b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WinMsiBundler.java index e159bef9d14..c26ee81da5e 100644 --- a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WinMsiBundler.java +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WinMsiBundler.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2021, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -82,7 +82,9 @@ import org.xml.sax.SAXException; *
  • main.wxs. Main source file with the installer description *
  • bundle.wxf. Source file with application and Java run-time directory tree * description. + *
  • ui.wxf. Source file with UI description of the installer. * + * *

    * main.wxs file is a copy of main.wxs resource from * jdk.jpackage.internal.resources package. It is parametrized with the @@ -104,15 +106,20 @@ import org.xml.sax.SAXException; * variable is set to the value of --win-upgrade-uuid command line option *

  • JpAllowDowngrades. Set to "yes" if --win-upgrade-uuid command line option * was specified. Undefined otherwise - *
  • JpLicenseRtf. Set to the value of --license-file command line option. - * Undefined is --license-file command line option was not specified - *
  • JpInstallDirChooser. Set to "yes" if --win-dir-chooser command line - * option was specified. Undefined otherwise *
  • JpConfigDir. Absolute path to the directory with generated WiX source * files. *
  • JpIsSystemWide. Set to "yes" if --win-per-user-install command line * option was not specified. Undefined otherwise * + * + *

    + * ui.wxf file is generated based on --license-file, --win-shortcut-prompt, + * --win-dir-chooser command line options. It is parametrized with the following + * WiX variables: + *

      + *
    • JpLicenseRtf. Set to the value of --license-file command line option. + * Undefined if --license-file command line option was not specified + *
    */ public class WinMsiBundler extends AbstractBundler { @@ -151,7 +158,6 @@ public class WinMsiBundler extends AbstractBundler { : Boolean.valueOf(s) ); - public static final StandardBundlerParam PRODUCT_VERSION = new StandardBundlerParam<>( "win.msi.productVersion", @@ -184,16 +190,15 @@ public class WinMsiBundler extends AbstractBundler { }, (s, p) -> s); - private static final BundlerParamInfo INSTALLDIR_CHOOSER = - new StandardBundlerParam<> ( - Arguments.CLIOptions.WIN_DIR_CHOOSER.getId(), - Boolean.class, - params -> Boolean.FALSE, - (s, p) -> Boolean.valueOf(s) - ); - public WinMsiBundler() { appImageBundler = new WinAppBundler().setDependentTask(true); + wixFragments = Stream.of( + Map.entry("bundle.wxf", new WixAppImageFragmentBuilder()), + Map.entry("ui.wxf", new WixUiFragmentBuilder()) + ).map(e -> { + e.getValue().setOutputFileName(e.getKey()); + return e.getValue(); + }).collect(Collectors.toList()); } @Override @@ -277,9 +282,10 @@ public class WinMsiBundler extends AbstractBundler { toolInfo.version)); } - wixSourcesBuilder.setWixVersion(wixToolset.get(WixTool.Light).version); + wixFragments.forEach(wixFragment -> wixFragment.setWixVersion( + wixToolset.get(WixTool.Light).version)); - wixSourcesBuilder.logWixFeatures(); + wixFragments.get(0).logWixFeatures(); /********* validate bundle parameters *************/ @@ -369,10 +375,10 @@ public class WinMsiBundler extends AbstractBundler { prepareProto(params); - wixSourcesBuilder - .initFromParams(WIN_APP_IMAGE.fetchFrom(params), params) - .createMainFragment(CONFIG_ROOT.fetchFrom(params).resolve( - "bundle.wxf")); + for (var wixFragment : wixFragments) { + wixFragment.initFromParams(params); + wixFragment.addFilesToConfigRoot(); + } Map wixVars = prepareMainProjectFile(params); @@ -424,22 +430,6 @@ public class WinMsiBundler extends AbstractBundler { data.put("JpIsSystemWide", "yes"); } - String licenseFile = LICENSE_FILE.fetchFrom(params); - if (licenseFile != null) { - String lname = IOUtils.getFileName(Path.of(licenseFile)).toString(); - Path destFile = CONFIG_ROOT.fetchFrom(params).resolve(lname); - data.put("JpLicenseRtf", destFile.toAbsolutePath().toString()); - } - - // Copy CA dll to include with installer - if (INSTALLDIR_CHOOSER.fetchFrom(params)) { - data.put("JpInstallDirChooser", "yes"); - String fname = "wixhelper.dll"; - try (InputStream is = OverridableResource.readDefault(fname)) { - Files.copy(is, CONFIG_ROOT.fetchFrom(params).resolve(fname)); - } - } - // Copy standard l10n files. for (String loc : Arrays.asList("en", "ja", "zh_CN")) { String fname = "MsiInstallerStrings_" + loc + ".wxl"; @@ -476,23 +466,20 @@ public class WinMsiBundler extends AbstractBundler { entry -> entry.getValue().path))) .setWixObjDir(TEMP_ROOT.fetchFrom(params).resolve("wixobj")) .setWorkDir(WIN_APP_IMAGE.fetchFrom(params)) - .addSource(CONFIG_ROOT.fetchFrom(params).resolve("main.wxs"), wixVars) - .addSource(CONFIG_ROOT.fetchFrom(params).resolve("bundle.wxf"), null); + .addSource(CONFIG_ROOT.fetchFrom(params).resolve("main.wxs"), wixVars); + + for (var wixFragment : wixFragments) { + wixFragment.configureWixPipeline(wixPipeline); + } Log.verbose(MessageFormat.format(I18N.getString( "message.generating-msi"), msiOut.toAbsolutePath().toString())); - boolean enableLicenseUI = (LICENSE_FILE.fetchFrom(params) != null); - boolean enableInstalldirUI = INSTALLDIR_CHOOSER.fetchFrom(params); - wixPipeline.addLightOptions("-sice:ICE27"); if (!MSI_SYSTEM_WIDE.fetchFrom(params)) { wixPipeline.addLightOptions("-sice:ICE91"); } - if (enableLicenseUI || enableInstalldirUI) { - wixPipeline.addLightOptions("-ext", "WixUIExtension"); - } final Path primaryWxlFile = CONFIG_ROOT.fetchFrom(params).resolve( I18N.getString("resource.wxl-file-name")).toAbsolutePath(); @@ -512,12 +499,6 @@ public class WinMsiBundler extends AbstractBundler { wixPipeline.addLightOptions(uniqueCultures.stream().collect( Collectors.joining(";", "-cultures:", ""))); - // Only needed if we using CA dll, so Wix can find it - if (enableInstalldirUI) { - wixPipeline.addLightOptions("-b", CONFIG_ROOT.fetchFrom(params) - .toAbsolutePath().toString()); - } - wixPipeline.buildMsi(msiOut.toAbsolutePath()); return msiOut; @@ -643,6 +624,6 @@ public class WinMsiBundler extends AbstractBundler { private Path installerIcon; private Map wixToolset; private AppImageBundler appImageBundler; - private WixSourcesBuilder wixSourcesBuilder = new WixSourcesBuilder(); + private List wixFragments; } diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixSourcesBuilder.java b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixAppImageFragmentBuilder.java similarity index 89% rename from src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixSourcesBuilder.java rename to src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixAppImageFragmentBuilder.java index 6c2dd9c5773..a9f8cd9aee1 100644 --- a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixSourcesBuilder.java +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixAppImageFragmentBuilder.java @@ -31,6 +31,7 @@ import java.nio.file.Path; import java.nio.file.Files; import java.text.MessageFormat; import java.util.ArrayList; +import java.util.Collection; import java.util.Collections; import java.util.Comparator; import java.util.HashMap; @@ -54,19 +55,19 @@ import static jdk.jpackage.internal.StandardBundlerParam.INSTALL_DIR; import static jdk.jpackage.internal.StandardBundlerParam.VENDOR; import static jdk.jpackage.internal.StandardBundlerParam.VERSION; import static jdk.jpackage.internal.WinMsiBundler.MSI_SYSTEM_WIDE; +import static jdk.jpackage.internal.WinMsiBundler.WIN_APP_IMAGE; /** - * Creates application WiX source files. + * Creates WiX fragment with components for contents of app image. */ -class WixSourcesBuilder { +class WixAppImageFragmentBuilder extends WixFragmentBuilder { - WixSourcesBuilder setWixVersion(DottedVersion v) { - wixVersion = v; - return this; - } + @Override + void initFromParams(Map params) { + super.initFromParams(params); + + Path appImageRoot = WIN_APP_IMAGE.fetchFrom(params); - WixSourcesBuilder initFromParams(Path appImageRoot, - Map params) { Supplier appImageSupplier = () -> { if (StandardBundlerParam.isRuntimeInstaller(params)) { return ApplicationLayout.javaRuntime(); @@ -101,64 +102,50 @@ class WixSourcesBuilder { installedAppImage = appImageSupplier.get().resolveAt(INSTALLDIR); - shortcutFolders = new HashSet<>(); - if (SHORTCUT_HINT.fetchFrom(params)) { - shortcutFolders.add(ShortcutsFolder.Desktop); - } - if (MENU_HINT.fetchFrom(params)) { - shortcutFolders.add(ShortcutsFolder.ProgramMenu); - } + shortcutFolders = Stream.of(ShortcutsFolder.values()).filter( + shortcutFolder -> shortcutFolder.requested(params)).collect( + Collectors.toSet()); if (StandardBundlerParam.isRuntimeInstaller(params)) { launcherPaths = Collections.emptyList(); } else { launcherPaths = AppImageFile.getLauncherNames(appImageRoot, params).stream() .map(name -> installedAppImage.launchersDirectory().resolve(name)) - .map(WixSourcesBuilder::addExeSuffixToPath) + .map(WixAppImageFragmentBuilder::addExeSuffixToPath) .collect(Collectors.toList()); } programMenuFolderName = MENU_GROUP.fetchFrom(params); initFileAssociations(params); - - return this; } - void createMainFragment(Path file) throws IOException { + @Override + void addFilesToConfigRoot() throws IOException { removeFolderItems = new HashMap<>(); defaultedMimes = new HashSet<>(); - IOUtils.createXml(file, xml -> { - xml.writeStartElement("Wix"); - xml.writeDefaultNamespace("http://schemas.microsoft.com/wix/2006/wi"); - xml.writeNamespace("util", - "http://schemas.microsoft.com/wix/UtilExtension"); - - xml.writeStartElement("Fragment"); - - addFaComponentGroup(xml); - - addShortcutComponentGroup(xml); - - addFilesComponentGroup(xml); - - xml.writeEndElement(); // - - addIconsFragment(xml); - - xml.writeEndElement(); // - }); + super.addFilesToConfigRoot(); } - void logWixFeatures() { - if (wixVersion.compareTo("3.6") >= 0) { - Log.verbose(MessageFormat.format(I18N.getString( - "message.use-wix36-features"), wixVersion)); - } - } + @Override + protected Collection getFragmentWriters() { + return List.of( + xml -> { + addFaComponentGroup(xml); - static boolean is64Bit() { - return !("x86".equals(System.getProperty("os.arch"))); + addShortcutComponentGroup(xml); + + addFilesComponentGroup(xml); + + for (var shortcutFolder : shortcutFolders) { + xml.writeStartElement("Property"); + xml.writeAttribute("Id", shortcutFolder.property); + xml.writeAttribute("Value", "1"); + xml.writeEndElement(); + } + }, + this::addIcons + ); } private void normalizeFileAssociation(FileAssociation fa) { @@ -365,6 +352,17 @@ class WixSourcesBuilder { Component.startElement(xml, componentId, String.format("{%s}", role.guidOf(path))); + if (role == Component.Shortcut) { + xml.writeStartElement("Condition"); + String property = shortcutFolders.stream().filter(shortcutFolder -> { + return path.startsWith(shortcutFolder.root); + }).map(shortcutFolder -> { + return shortcutFolder.property; + }).findFirst().get(); + xml.writeCharacters(property); + xml.writeEndElement(); + } + boolean isRegistryKeyPath = !systemWide || role.isRegistryKeyPath(); if (isRegistryKeyPath) { addRegistryKeyPath(xml, directoryRefPath); @@ -659,7 +657,7 @@ class WixSourcesBuilder { addComponentGroup(xml, "Files", componentIds); } - private void addIconsFragment(XMLStreamWriter xml) throws + private void addIcons(XMLStreamWriter xml) throws XMLStreamException, IOException { PathGroup srcPathGroup = appImage.pathGroup(); @@ -680,14 +678,12 @@ class WixSourcesBuilder { } }); - xml.writeStartElement("Fragment"); for (var icoFile : icoFiles) { xml.writeStartElement("Icon"); xml.writeAttribute("Id", Id.Icon.of(icoFile.getValue())); xml.writeAttribute("SourceFile", icoFile.getKey().toString()); xml.writeEndElement(); } - xml.writeEndElement(); } private void addRegistryKeyPath(XMLStreamWriter xml, Path path) throws @@ -705,7 +701,7 @@ class WixSourcesBuilder { xml.writeStartElement("RegistryKey"); xml.writeAttribute("Root", regRoot); xml.writeAttribute("Key", registryKeyPath); - if (wixVersion.compareTo("3.6") < 0) { + if (getWixVersion().compareTo("3.6") < 0) { xml.writeAttribute("Action", "createAndRemoveOnUninstall"); } xml.writeStartElement("RegistryValue"); @@ -719,7 +715,7 @@ class WixSourcesBuilder { private String addDirectoryCleaner(XMLStreamWriter xml, Path path) throws XMLStreamException, IOException { - if (wixVersion.compareTo("3.6") < 0) { + if (getWixVersion().compareTo("3.6") < 0) { return null; } @@ -775,24 +771,46 @@ class WixSourcesBuilder { } enum ShortcutsFolder { - ProgramMenu(PROGRAM_MENU_PATH), - Desktop(DESKTOP_PATH); + ProgramMenu(PROGRAM_MENU_PATH, Arguments.CLIOptions.WIN_MENU_HINT, + "JP_INSTALL_STARTMENU_SHORTCUT", "JpStartMenuShortcutPrompt"), + Desktop(DESKTOP_PATH, Arguments.CLIOptions.WIN_SHORTCUT_HINT, + "JP_INSTALL_DESKTOP_SHORTCUT", "JpDesktopShortcutPrompt"); - private ShortcutsFolder(Path root) { + private ShortcutsFolder(Path root, Arguments.CLIOptions cliOption, + String property, String wixVariableName) { this.root = root; + this.bundlerParam = new StandardBundlerParam<>( + cliOption.getId(), + Boolean.class, + params -> false, + // valueOf(null) is false, + // and we actually do want null in some cases + (s, p) -> (s == null || "null".equalsIgnoreCase(s)) ? false : Boolean.valueOf(s) + ); + this.wixVariableName = wixVariableName; + this.property = property; } - Path getPath(WixSourcesBuilder outer) { + Path getPath(WixAppImageFragmentBuilder outer) { if (this == ProgramMenu) { return root.resolve(outer.programMenuFolderName); } return root; } - private final Path root; - } + boolean requested(Map params) { + return bundlerParam.fetchFrom(params); + } - private DottedVersion wixVersion; + String getWixVariableName() { + return wixVariableName; + } + + private final Path root; + private final String property; + private final String wixVariableName; + private final StandardBundlerParam bundlerParam; + } private boolean systemWide; @@ -839,28 +857,6 @@ class WixSourcesBuilder { private final static Set USER_PROFILE_DIRS = Set.of(LOCAL_PROGRAM_FILES, PROGRAM_MENU_PATH, DESKTOP_PATH); - private static final StandardBundlerParam MENU_HINT = - new StandardBundlerParam<>( - Arguments.CLIOptions.WIN_MENU_HINT.getId(), - Boolean.class, - params -> false, - // valueOf(null) is false, - // and we actually do want null in some cases - (s, p) -> (s == null || - "null".equalsIgnoreCase(s))? true : Boolean.valueOf(s) - ); - - private static final StandardBundlerParam SHORTCUT_HINT = - new StandardBundlerParam<>( - Arguments.CLIOptions.WIN_SHORTCUT_HINT.getId(), - Boolean.class, - params -> false, - // valueOf(null) is false, - // and we actually do want null in some cases - (s, p) -> (s == null || - "null".equalsIgnoreCase(s))? false : Boolean.valueOf(s) - ); - private static final StandardBundlerParam MENU_GROUP = new StandardBundlerParam<>( Arguments.CLIOptions.WIN_MENU_GROUP.getId(), diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixFragmentBuilder.java b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixFragmentBuilder.java new file mode 100644 index 00000000000..dacf7470b7d --- /dev/null +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixFragmentBuilder.java @@ -0,0 +1,156 @@ +/* + * Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package jdk.jpackage.internal; + +import java.io.IOException; +import java.nio.file.Path; +import java.text.MessageFormat; +import java.util.ArrayList; +import java.util.Collection; +import java.util.List; +import java.util.Map; +import java.util.Optional; +import jdk.jpackage.internal.IOUtils.XmlConsumer; +import jdk.jpackage.internal.OverridableResource.Source; +import static jdk.jpackage.internal.OverridableResource.createResource; +import static jdk.jpackage.internal.StandardBundlerParam.CONFIG_ROOT; + +/** + * Creates WiX fragment. + */ +abstract class WixFragmentBuilder { + + void setWixVersion(DottedVersion v) { + wixVersion = v; + } + + void setOutputFileName(String v) { + outputFileName = v; + } + + void initFromParams(Map params) { + wixVariables = null; + additionalResources = null; + configRoot = CONFIG_ROOT.fetchFrom(params); + fragmentResource = createResource(outputFileName, params).setSourceOrder( + Source.ResourceDir); + } + + void logWixFeatures() { + if (wixVersion.compareTo("3.6") >= 0) { + Log.verbose(MessageFormat.format(I18N.getString( + "message.use-wix36-features"), wixVersion)); + } + } + + void configureWixPipeline(WixPipeline wixPipeline) { + wixPipeline.addSource(configRoot.resolve(outputFileName), + Optional.ofNullable(wixVariables).map(WixVariables::getValues).orElse( + null)); + } + + void addFilesToConfigRoot() throws IOException { + Path fragmentPath = configRoot.resolve(outputFileName); + if (fragmentResource.saveToFile(fragmentPath) == null) { + createWixSource(fragmentPath, xml -> { + for (var fragmentWriter : getFragmentWriters()) { + xml.writeStartElement("Fragment"); + fragmentWriter.accept(xml); + xml.writeEndElement(); // + } + }); + } + + if (additionalResources != null) { + for (var resource : additionalResources) { + resource.resource.saveToFile(configRoot.resolve( + resource.saveAsName)); + } + } + } + + DottedVersion getWixVersion() { + return wixVersion; + } + + static boolean is64Bit() { + return !("x86".equals(System.getProperty("os.arch"))); + } + + protected Path getConfigRoot() { + return configRoot; + } + + protected abstract Collection getFragmentWriters(); + + protected void defineWixVariable(String variableName) { + setWixVariable(variableName, "yes"); + } + + protected void setWixVariable(String variableName, String variableValue) { + if (wixVariables == null) { + wixVariables = new WixVariables(); + } + wixVariables.setWixVariable(variableName, variableValue); + } + + protected void addResource(OverridableResource resource, String saveAsName) { + if (additionalResources == null) { + additionalResources = new ArrayList<>(); + } + additionalResources.add(new ResourceWithName(resource, saveAsName)); + } + + static void createWixSource(Path file, XmlConsumer xmlConsumer) + throws IOException { + IOUtils.createXml(file, xml -> { + xml.writeStartElement("Wix"); + xml.writeDefaultNamespace("http://schemas.microsoft.com/wix/2006/wi"); + xml.writeNamespace("util", + "http://schemas.microsoft.com/wix/UtilExtension"); + + xmlConsumer.accept(xml); + + xml.writeEndElement(); // + }); + } + + private static class ResourceWithName { + + ResourceWithName(OverridableResource resource, String saveAsName) { + this.resource = resource; + this.saveAsName = saveAsName; + } + private final OverridableResource resource; + private final String saveAsName; + } + + private DottedVersion wixVersion; + private WixVariables wixVariables; + private List additionalResources; + private OverridableResource fragmentResource; + private String outputFileName; + private Path configRoot; +} diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixPipeline.java b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixPipeline.java index 586ec02dddf..19f319ed9c2 100644 --- a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixPipeline.java +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixPipeline.java @@ -112,7 +112,7 @@ public class WixPipeline { "-nologo", adjustPath.apply(wixSource.source).toString(), "-ext", "WixUtilExtension", - "-arch", WixSourcesBuilder.is64Bit() ? "x64" : "x86", + "-arch", WixFragmentBuilder.is64Bit() ? "x64" : "x86", "-out", wixObj.toAbsolutePath().toString() )); diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixUiFragmentBuilder.java b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixUiFragmentBuilder.java new file mode 100644 index 00000000000..2211994bbad --- /dev/null +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixUiFragmentBuilder.java @@ -0,0 +1,501 @@ +/* + * Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package jdk.jpackage.internal; + +import java.io.IOException; +import java.io.InputStream; +import java.nio.file.Files; +import java.nio.file.Path; +import java.util.ArrayList; +import java.util.Collection; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Objects; +import java.util.function.Function; +import java.util.function.Supplier; +import java.util.stream.Collectors; +import java.util.stream.Stream; +import javax.xml.stream.XMLStreamException; +import javax.xml.stream.XMLStreamWriter; +import jdk.jpackage.internal.IOUtils.XmlConsumer; +import static jdk.jpackage.internal.OverridableResource.createResource; +import static jdk.jpackage.internal.StandardBundlerParam.LICENSE_FILE; +import jdk.jpackage.internal.WixAppImageFragmentBuilder.ShortcutsFolder; + +/** + * Creates UI WiX fragment. + */ +final class WixUiFragmentBuilder extends WixFragmentBuilder { + + @Override + void initFromParams(Map params) { + super.initFromParams(params); + + String licenseFile = LICENSE_FILE.fetchFrom(params); + withLicenseDlg = licenseFile != null; + if (withLicenseDlg) { + Path licenseFileName = IOUtils.getFileName(Path.of(licenseFile)); + Path destFile = getConfigRoot().resolve(licenseFileName); + setWixVariable("JpLicenseRtf", destFile.toAbsolutePath().toString()); + } + + withInstallDirChooserDlg = INSTALLDIR_CHOOSER.fetchFrom(params); + + List shortcutFolders = Stream.of( + ShortcutsFolder.values()).filter(shortcutFolder -> { + return shortcutFolder.requested(params) + && SHORTCUT_PROMPT.fetchFrom(params); + }).collect(Collectors.toList()); + + withShortcutPromptDlg = !shortcutFolders.isEmpty(); + + customDialogs = new ArrayList<>(); + + if (withShortcutPromptDlg) { + CustomDialog dialog = new CustomDialog(params, I18N.getString( + "resource.shortcutpromptdlg-wix-file"), + "ShortcutPromptDlg.wxs"); + for (var shortcutFolder : shortcutFolders) { + dialog.wixVariables.defineWixVariable( + shortcutFolder.getWixVariableName()); + } + customDialogs.add(dialog); + } + + if (withInstallDirChooserDlg) { + CustomDialog dialog = new CustomDialog(params, I18N.getString( + "resource.installdirnotemptydlg-wix-file"), + "InstallDirNotEmptyDlg.wxs"); + List dialogIds = getUI().dialogIdsSupplier.apply(this); + dialog.wixVariables.setWixVariable("JpAfterInstallDirDlg", + dialogIds.get(dialogIds.indexOf(Dialog.InstallDirDlg) + 1).id); + customDialogs.add(dialog); + } + } + + @Override + void configureWixPipeline(WixPipeline wixPipeline) { + super.configureWixPipeline(wixPipeline); + + if (withShortcutPromptDlg || withInstallDirChooserDlg || withLicenseDlg) { + wixPipeline.addLightOptions("-ext", "WixUIExtension"); + } + + // Only needed if we using CA dll, so Wix can find it + if (withInstallDirChooserDlg) { + wixPipeline.addLightOptions("-b", + getConfigRoot().toAbsolutePath().toString()); + } + + for (var customDialog : customDialogs) { + customDialog.addToWixPipeline(wixPipeline); + } + } + + @Override + void addFilesToConfigRoot() throws IOException { + super.addFilesToConfigRoot(); + + if (withInstallDirChooserDlg) { + String fname = "wixhelper.dll"; // CA dll + try (InputStream is = OverridableResource.readDefault(fname)) { + Files.copy(is, getConfigRoot().resolve(fname)); + } + } + } + + @Override + protected Collection getFragmentWriters() { + return List.of(this::addUI); + } + + private void addUI(XMLStreamWriter xml) throws XMLStreamException, + IOException { + + if (withInstallDirChooserDlg) { + xml.writeStartElement("Property"); + xml.writeAttribute("Id", "WIXUI_INSTALLDIR"); + xml.writeAttribute("Value", "INSTALLDIR"); + xml.writeEndElement(); // Property + } + + if (withLicenseDlg) { + xml.writeStartElement("WixVariable"); + xml.writeAttribute("Id", "WixUILicenseRtf"); + xml.writeAttribute("Value", "$(var.JpLicenseRtf)"); + xml.writeEndElement(); // WixVariable + } + + xml.writeStartElement("UI"); + xml.writeAttribute("Id", "JpUI"); + + var ui = getUI(); + if (ui != null) { + ui.write(this, xml); + } + + xml.writeEndElement(); // UI + } + + private UI getUI() { + if (withInstallDirChooserDlg || withShortcutPromptDlg) { + // WixUI_InstallDir for shortcut prompt dialog too because in + // WixUI_Minimal UI sequence WelcomeEulaDlg dialog doesn't have "Next" + // button, but has "Install" button. So inserting shortcut prompt dialog + // after welcome dialog in WixUI_Minimal UI sequence would be confusing + return UI.InstallDir; + } else if (withLicenseDlg) { + return UI.Minimal; + } else { + return null; + } + } + + private enum UI { + InstallDir("WixUI_InstallDir", + WixUiFragmentBuilder::dialogSequenceForWixUI_InstallDir, + Dialog::createPairsForWixUI_InstallDir), + Minimal("WixUI_Minimal", null, null); + + UI(String wixUIRef, + Function> dialogIdsSupplier, + Supplier>> dialogPairsSupplier) { + this.wixUIRef = wixUIRef; + this.dialogIdsSupplier = dialogIdsSupplier; + this.dialogPairsSupplier = dialogPairsSupplier; + } + + void write(WixUiFragmentBuilder outer, XMLStreamWriter xml) throws + XMLStreamException, IOException { + xml.writeStartElement("UIRef"); + xml.writeAttribute("Id", wixUIRef); + xml.writeEndElement(); // UIRef + + if (dialogIdsSupplier != null) { + List dialogIds = dialogIdsSupplier.apply(outer); + Map> dialogPairs = dialogPairsSupplier.get(); + + if (dialogIds.contains(Dialog.InstallDirDlg)) { + xml.writeStartElement("DialogRef"); + xml.writeAttribute("Id", "InstallDirNotEmptyDlg"); + xml.writeEndElement(); // DialogRef + } + + var it = dialogIds.iterator(); + Dialog firstId = it.next(); + while (it.hasNext()) { + Dialog secondId = it.next(); + DialogPair pair = new DialogPair(firstId, secondId); + for (var curPair : List.of(pair, pair.flip())) { + for (var publish : dialogPairs.get(curPair)) { + writePublishDialogPair(xml, publish, curPair); + } + } + firstId = secondId; + } + } + } + + private final String wixUIRef; + private final Function> dialogIdsSupplier; + private final Supplier>> dialogPairsSupplier; + } + + private List dialogSequenceForWixUI_InstallDir() { + List dialogIds = new ArrayList<>( + List.of(Dialog.WixUI_WelcomeDlg)); + if (withLicenseDlg) { + dialogIds.add(Dialog.WixUI_LicenseAgreementDlg); + } + + if (withInstallDirChooserDlg) { + dialogIds.add(Dialog.InstallDirDlg); + } + + if (withShortcutPromptDlg) { + dialogIds.add(Dialog.ShortcutPromptDlg); + } + + dialogIds.add(Dialog.WixUI_VerifyReadyDlg); + + return dialogIds; + } + + private enum Dialog { + WixUI_WelcomeDlg, + WixUI_LicenseAgreementDlg, + InstallDirDlg, + ShortcutPromptDlg, + WixUI_VerifyReadyDlg; + + Dialog() { + if (name().startsWith("WixUI_")) { + id = name().substring("WixUI_".length()); + } else { + id = name(); + } + } + + static Map> createPair(Dialog firstId, + Dialog secondId, List nextBuilders, + List prevBuilders) { + var pair = new DialogPair(firstId, secondId); + return Map.of(pair, nextBuilders.stream().map(b -> { + return buildPublish(b.create()).next().create(); + }).collect(Collectors.toList()), pair.flip(), + prevBuilders.stream().map(b -> { + return buildPublish(b.create()).back().create(); + }).collect(Collectors.toList())); + } + + static Map> createPair(Dialog firstId, + Dialog secondId, List builders) { + return createPair(firstId, secondId, builders, builders); + } + + static Map> createPairsForWixUI_InstallDir() { + Map> map = new HashMap<>(); + + // Order is a "weight" of action. If there are multiple + // "NewDialog" action for the same dialog Id, MSI would pick the one + // with higher order value. In WixUI_InstallDir dialog sequence the + // highest order value is 4. InstallDirNotEmptyDlg adds NewDialog + // action with order 5. Setting order to 6 for all + // actions configured in this function would make them executed + // instead of corresponding default actions defined in + // WixUI_InstallDir dialog sequence. + var order = 6; + + // Based on WixUI_InstallDir.wxs + var backFromVerifyReadyDlg = List.of(buildPublish().condition( + "NOT Installed").order(order)); + var uncondinal = List.of(buildPublish().condition("1")); + var ifNotIstalled = List.of( + buildPublish().condition("NOT Installed").order(order)); + var ifLicenseAccepted = List.of(buildPublish().condition( + "LicenseAccepted = \"1\"").order(order)); + + // Empty condition list for the default dialog sequence + map.putAll(createPair(WixUI_WelcomeDlg, WixUI_LicenseAgreementDlg, + List.of())); + map.putAll( + createPair(WixUI_WelcomeDlg, InstallDirDlg, ifNotIstalled)); + map.putAll(createPair(WixUI_WelcomeDlg, ShortcutPromptDlg, + ifNotIstalled)); + + map.putAll(createPair(WixUI_LicenseAgreementDlg, InstallDirDlg, + List.of())); + map.putAll(createPair(WixUI_LicenseAgreementDlg, ShortcutPromptDlg, + ifLicenseAccepted, uncondinal)); + map.putAll(createPair(WixUI_LicenseAgreementDlg, + WixUI_VerifyReadyDlg, ifLicenseAccepted, + backFromVerifyReadyDlg)); + + map.putAll(createPair(InstallDirDlg, ShortcutPromptDlg, List.of(), + uncondinal)); + map.putAll(createPair(InstallDirDlg, WixUI_VerifyReadyDlg, List.of())); + + map.putAll(createPair(ShortcutPromptDlg, WixUI_VerifyReadyDlg, + uncondinal, backFromVerifyReadyDlg)); + + return map; + } + + private final String id; + } + + private final static class DialogPair { + + DialogPair(Dialog first, Dialog second) { + this(first.id, second.id); + } + + DialogPair(String firstId, String secondId) { + this.firstId = firstId; + this.secondId = secondId; + } + + DialogPair flip() { + return new DialogPair(secondId, firstId); + } + + @Override + public int hashCode() { + int hash = 3; + hash = 97 * hash + Objects.hashCode(this.firstId); + hash = 97 * hash + Objects.hashCode(this.secondId); + return hash; + } + + @Override + public boolean equals(Object obj) { + if (this == obj) { + return true; + } + if (obj == null) { + return false; + } + if (getClass() != obj.getClass()) { + return false; + } + final DialogPair other = (DialogPair) obj; + if (!Objects.equals(this.firstId, other.firstId)) { + return false; + } + if (!Objects.equals(this.secondId, other.secondId)) { + return false; + } + return true; + } + + private final String firstId; + private final String secondId; + } + + private final static class Publish { + + Publish(String control, String condition, int order) { + this.control = control; + this.condition = condition; + this.order = order; + } + + private final String control; + private final String condition; + private final int order; + } + + private final static class PublishBuilder { + + PublishBuilder() { + order(0); + next(); + condition("1"); + } + + PublishBuilder(Publish publish) { + order(publish.order); + control(publish.control); + condition(publish.condition); + } + + public PublishBuilder control(String v) { + control = v; + return this; + } + + public PublishBuilder next() { + return control("Next"); + } + + public PublishBuilder back() { + return control("Back"); + } + + public PublishBuilder condition(String v) { + condition = v; + return this; + } + + public PublishBuilder order(int v) { + order = v; + return this; + } + + Publish create() { + return new Publish(control, condition, order); + } + + private String control; + private String condition; + private int order; + } + + private static PublishBuilder buildPublish() { + return new PublishBuilder(); + } + + private static PublishBuilder buildPublish(Publish publish) { + return new PublishBuilder(publish); + } + + private static void writePublishDialogPair(XMLStreamWriter xml, + Publish publish, DialogPair dialogPair) throws IOException, + XMLStreamException { + xml.writeStartElement("Publish"); + xml.writeAttribute("Dialog", dialogPair.firstId); + xml.writeAttribute("Control", publish.control); + xml.writeAttribute("Event", "NewDialog"); + xml.writeAttribute("Value", dialogPair.secondId); + if (publish.order != 0) { + xml.writeAttribute("Order", String.valueOf(publish.order)); + } + xml.writeCharacters(publish.condition); + xml.writeEndElement(); + } + + private final class CustomDialog { + + CustomDialog(Map params, String category, + String wxsFileName) { + this.wxsFileName = wxsFileName; + this.wixVariables = new WixVariables(); + + addResource( + createResource(wxsFileName, params).setCategory(category), + wxsFileName); + } + + void addToWixPipeline(WixPipeline wixPipeline) { + wixPipeline.addSource(getConfigRoot().toAbsolutePath().resolve( + wxsFileName), wixVariables.getValues()); + } + + private final WixVariables wixVariables; + private final String wxsFileName; + } + + private boolean withInstallDirChooserDlg; + private boolean withShortcutPromptDlg; + private boolean withLicenseDlg; + private List customDialogs; + + private static final BundlerParamInfo INSTALLDIR_CHOOSER + = new StandardBundlerParam<>( + Arguments.CLIOptions.WIN_DIR_CHOOSER.getId(), + Boolean.class, + params -> false, + (s, p) -> Boolean.valueOf(s) + ); + + private static final StandardBundlerParam SHORTCUT_PROMPT + = new StandardBundlerParam<>( + Arguments.CLIOptions.WIN_SHORTCUT_PROMPT.getId(), + Boolean.class, + params -> false, + (s, p) -> Boolean.valueOf(s) + ); +} diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixVariables.java b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixVariables.java new file mode 100644 index 00000000000..36ed1d99738 --- /dev/null +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/WixVariables.java @@ -0,0 +1,45 @@ +/* + * Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package jdk.jpackage.internal; + +import java.util.HashMap; +import java.util.Map; + +final class WixVariables { + + void defineWixVariable(String variableName) { + setWixVariable(variableName, "yes"); + } + + void setWixVariable(String variableName, String variableValue) { + values.put(variableName, variableValue); + } + + Map getValues() { + return values; + } + + private final Map values = new HashMap<>(); +} diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/InstallDirNotEmptyDlg.wxs b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/InstallDirNotEmptyDlg.wxs new file mode 100644 index 00000000000..a477a8b6330 --- /dev/null +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/InstallDirNotEmptyDlg.wxs @@ -0,0 +1,56 @@ + + + + + + + + + + + + + + 1 + + + 1 + + + !(loc.message.install.dir.exist) + + + + 1 + INSTALLDIR_VALID="0" + INSTALLDIR_VALID="1" + + + + diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/MsiInstallerStrings_en.wxl b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/MsiInstallerStrings_en.wxl index 9e88d1d10f1..4622cca5d54 100644 --- a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/MsiInstallerStrings_en.wxl +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/MsiInstallerStrings_en.wxl @@ -4,4 +4,12 @@ Main Feature A higher version of [ProductName] is already installed. Downgrades disabled. Setup will now exit. A lower version of [ProductName] is already installed. Upgrades disabled. Setup will now exit. + + [ProductName] Setup + {\WixUI_Font_Title}Shortcuts + WixUI_Bmp_Banner + Select shortcuts to create. + Create desktop shortcut(s) + Create start menu shortcut(s) + [ProductName] Setup diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/MsiInstallerStrings_ja.wxl b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/MsiInstallerStrings_ja.wxl index 61037204d12..a8403e5ba48 100644 --- a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/MsiInstallerStrings_ja.wxl +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/MsiInstallerStrings_ja.wxl @@ -4,4 +4,12 @@ 主な機能 [ProductName]のより上位のバージョンがすでにインストールされています。ダウングレードは無効です。セットアップを終了します。 [ProductName]のより下位のバージョンがすでにインストールされています。アップグレードは無効です。セットアップを終了します。 + + [ProductName] Setup + {\WixUI_Font_Title}Shortcuts + WixUI_Bmp_Banner + Select shortcuts to create. + Create desktop shortcut(s) + Create start menu shortcut(s) + [ProductName] Setup diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/MsiInstallerStrings_zh_CN.wxl b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/MsiInstallerStrings_zh_CN.wxl index 63e28b05a35..ca52140fff4 100644 --- a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/MsiInstallerStrings_zh_CN.wxl +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/MsiInstallerStrings_zh_CN.wxl @@ -4,4 +4,12 @@ 主要功能 已安装更高版本的 [ProductName]。降级已禁用。现在将退出安装。 已安装更低版本的 [ProductName]。升级已禁用。现在将退出安装。 + + [ProductName] Setup + {\WixUI_Font_Title}Shortcuts + WixUI_Bmp_Banner + Select shortcuts to create. + Create desktop shortcut(s) + Create start menu shortcut(s) + [ProductName] Setup diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/ShortcutPromptDlg.wxs b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/ShortcutPromptDlg.wxs new file mode 100644 index 00000000000..33f7b484328 --- /dev/null +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/ShortcutPromptDlg.wxs @@ -0,0 +1,78 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 1 + + + + + diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/WinResources.properties b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/WinResources.properties index af69415fdb0..e335fc35bf0 100644 --- a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/WinResources.properties +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/WinResources.properties @@ -1,5 +1,5 @@ # -# Copyright (c) 2017, 2020, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2017, 2021, Oracle and/or its affiliates. All rights reserved. # DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. # # This code is free software; you can redistribute it and/or modify it @@ -37,6 +37,8 @@ resource.post-msi-script=script to run after msi file for exe installer is creat resource.wxl-file-name=MsiInstallerStrings_en.wxl resource.main-wix-file=Main WiX project file resource.overrides-wix-file=Overrides WiX project file +resource.shortcutpromptdlg-wix-file=Shortcut prompt dialog WiX project file +resource.installdirnotemptydlg-wix-file=Not empty install directory dialog WiX project file error.no-wix-tools=Can not find WiX tools (light.exe, candle.exe) error.no-wix-tools.advice=Download WiX 3.0 or later from https://wixtoolset.org and add it to the PATH. diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/WinResources_ja.properties b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/WinResources_ja.properties index a8a9e5504d7..027f32ee2a8 100644 --- a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/WinResources_ja.properties +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/WinResources_ja.properties @@ -37,6 +37,8 @@ resource.post-msi-script=exe\u30A4\u30F3\u30B9\u30C8\u30FC\u30E9\u306Emsi\u30D5\ resource.wxl-file-name=MsiInstallerStrings_ja.wxl resource.main-wix-file=\u30E1\u30A4\u30F3WiX\u30D7\u30ED\u30B8\u30A7\u30AF\u30C8\u30FB\u30D5\u30A1\u30A4\u30EB resource.overrides-wix-file=WiX\u30D7\u30ED\u30B8\u30A7\u30AF\u30C8\u30FB\u30D5\u30A1\u30A4\u30EB\u306E\u30AA\u30FC\u30D0\u30FC\u30E9\u30A4\u30C9 +resource.shortcutpromptdlg-wix-file=Shortcut prompt dialog WiX project file +resource.installdirnotemptydlg-wix-file=Not empty install directory dialog WiX project file error.no-wix-tools=WiX\u30C4\u30FC\u30EB(light.exe\u3001candle.exe)\u304C\u898B\u3064\u304B\u308A\u307E\u305B\u3093 error.no-wix-tools.advice=WiX 3.0\u4EE5\u964D\u3092https://wixtoolset.org\u304B\u3089\u30C0\u30A6\u30F3\u30ED\u30FC\u30C9\u3057\u3001PATH\u306B\u8FFD\u52A0\u3057\u307E\u3059\u3002 diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/WinResources_zh_CN.properties b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/WinResources_zh_CN.properties index db020f1fc78..7eb96b1319c 100644 --- a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/WinResources_zh_CN.properties +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/WinResources_zh_CN.properties @@ -37,6 +37,8 @@ resource.post-msi-script=\u5728\u4E3A exe \u5B89\u88C5\u7A0B\u5E8F\u521B\u5EFA m resource.wxl-file-name=MsiInstallerStrings_zh_CN.wxl resource.main-wix-file=\u4E3B WiX \u9879\u76EE\u6587\u4EF6 resource.overrides-wix-file=\u8986\u76D6 WiX \u9879\u76EE\u6587\u4EF6 +resource.shortcutpromptdlg-wix-file=Shortcut prompt dialog WiX project file +resource.installdirnotemptydlg-wix-file=Not empty install directory dialog WiX project file error.no-wix-tools=\u627E\u4E0D\u5230 WiX \u5DE5\u5177 (light.exe, candle.exe) error.no-wix-tools.advice=\u4ECE https://wixtoolset.org \u4E0B\u8F7D WiX 3.0 \u6216\u66F4\u9AD8\u7248\u672C\uFF0C\u7136\u540E\u5C06\u5176\u6DFB\u52A0\u5230 PATH\u3002 diff --git a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/main.wxs b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/main.wxs index b7b65d7b61b..d2d0b55aea7 100644 --- a/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/main.wxs +++ b/src/jdk.jpackage/windows/classes/jdk/jpackage/internal/resources/main.wxs @@ -74,64 +74,14 @@ - - - - - - - - - - - 1 - - - 1 - - - !(loc.message.install.dir.exist) - - - - - - - - 1 - INSTALLDIR_VALID="0" - INSTALLDIR_VALID="1" - - - - 1 - 1 - - - - - - - - - - - - - - + + Not Installed diff --git a/test/jdk/tools/jpackage/windows/WinInstallerUiTest.java b/test/jdk/tools/jpackage/windows/WinInstallerUiTest.java new file mode 100644 index 00000000000..2e287cafe68 --- /dev/null +++ b/test/jdk/tools/jpackage/windows/WinInstallerUiTest.java @@ -0,0 +1,124 @@ +/* + * Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +import java.nio.file.Path; +import java.util.ArrayList; +import jdk.jpackage.test.PackageTest; +import jdk.jpackage.test.JPackageCommand; +import jdk.jpackage.test.Annotations.Test; +import jdk.jpackage.test.Annotations.Parameters; +import java.util.List; +import jdk.jpackage.test.PackageType; +import jdk.jpackage.test.TKit; + +/** + * Test all possible combinations of --win-dir-chooser, --win-shortcut-prompt + * and --license parameters. + */ + +/* + * @test + * @summary jpackage with --win-dir-chooser, --win-shortcut-prompt and --license parameters + * @library ../helpers + * @key jpackagePlatformPackage + * @build jdk.jpackage.test.* + * @requires (os.family == "windows") + * @modules jdk.jpackage/jdk.jpackage.internal + * @run main/othervm/timeout=360 -Xmx512m jdk.jpackage.test.Main + * --jpt-run=WinInstallerUiTest + */ +public class WinInstallerUiTest { + + public WinInstallerUiTest(Boolean withDirChooser, Boolean withLicense, + Boolean withShortcutPrompt) { + this.withShortcutPrompt = withShortcutPrompt; + this.withDirChooser = withDirChooser; + this.withLicense = withLicense; + } + + @Parameters + public static List data() { + List data = new ArrayList<>(); + for (var withDirChooser : List.of(Boolean.TRUE, Boolean.FALSE)) { + for (var withLicense : List.of(Boolean.TRUE, Boolean.FALSE)) { + for (var withShortcutPrompt : List.of(Boolean.TRUE, Boolean.FALSE)) { + data.add(new Object[]{withDirChooser, withLicense, + withShortcutPrompt}); + } + } + } + + return data; + } + + @Test + public void test() { + PackageTest test = new PackageTest() + .forTypes(PackageType.WINDOWS) + .configureHelloApp(); + + test.addInitializer(JPackageCommand::setFakeRuntime); + test.addInitializer(this::setPackageName); + + if (withDirChooser) { + test.addInitializer(cmd -> cmd.addArgument("--win-dir-chooser")); + } + + if (withShortcutPrompt) { + test.addInitializer(cmd -> { + cmd.addArgument("--win-shortcut-prompt"); + cmd.addArgument("--win-menu"); + cmd.addArgument("--win-shortcut"); + }); + } + + if (withLicense) { + test.addInitializer(cmd -> { + cmd.addArguments("--license-file", TKit.createRelativePathCopy( + TKit.TEST_SRC_ROOT.resolve(Path.of("resources", + "license.txt")))); + }); + } + + test.run(); + } + + private void setPackageName(JPackageCommand cmd) { + StringBuilder sb = new StringBuilder(cmd.name()); + sb.append("With"); + if (withDirChooser) { + sb.append("DirChooser"); + } + if (withShortcutPrompt) { + sb.append("ShortcutPrompt"); + } + if (withLicense) { + sb.append("License"); + } + cmd.setArgumentValue("--name", sb.toString()); + } + + private final boolean withDirChooser; + private final boolean withLicense; + private final boolean withShortcutPrompt; +} diff --git a/test/jdk/tools/jpackage/windows/WinShortcutPromptTest.java b/test/jdk/tools/jpackage/windows/WinShortcutPromptTest.java new file mode 100644 index 00000000000..e439cbf88a8 --- /dev/null +++ b/test/jdk/tools/jpackage/windows/WinShortcutPromptTest.java @@ -0,0 +1,114 @@ +/* + * Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +import java.util.ArrayList; +import jdk.jpackage.test.PackageTest; +import jdk.jpackage.test.JPackageCommand; +import jdk.jpackage.test.Annotations.Test; +import jdk.jpackage.test.Annotations.Parameters; +import java.util.List; +import jdk.jpackage.test.PackageType; + +/** + * Test all possible combinations of --win-shortcut-prompt, --win-menu and + * --win-shortcut parameters. + */ + +/* + * @test + * @summary jpackage with --win-shortcut-prompt, --win-menu and --win-shortcut parameters + * @library ../helpers + * @key jpackagePlatformPackage + * @build jdk.jpackage.test.* + * @requires (os.family == "windows") + * @modules jdk.jpackage/jdk.jpackage.internal + * @run main/othervm/timeout=360 -Xmx512m jdk.jpackage.test.Main + * --jpt-run=WinShortcutPromptTest + */ +public class WinShortcutPromptTest { + + public WinShortcutPromptTest(Boolean withStartMenuShortcut, + Boolean withDesktopShortcut, Boolean withShortcutPrompt) { + this.withStartMenuShortcut = withStartMenuShortcut; + this.withDesktopShortcut = withDesktopShortcut; + this.withShortcutPrompt = withShortcutPrompt; + } + + @Parameters + public static List data() { + List data = new ArrayList<>(); + for (var withStartMenuShortcut : List.of(Boolean.TRUE, Boolean.FALSE)) { + for (var withDesktopShortcut : List.of(Boolean.TRUE, Boolean.FALSE)) { + for (var withShortcutPrompt : List.of(Boolean.TRUE, Boolean.FALSE)) { + data.add(new Object[]{withStartMenuShortcut, + withDesktopShortcut, withShortcutPrompt}); + } + } + } + + return data; + } + + @Test + public void test() { + PackageTest test = new PackageTest() + .forTypes(PackageType.WINDOWS) + .configureHelloApp(); + + test.addInitializer(JPackageCommand::setFakeRuntime); + test.addInitializer(this::setPackageName); + + if (withShortcutPrompt) { + test.addInitializer(cmd -> cmd.addArgument("--win-shortcut-prompt")); + } + + if (withStartMenuShortcut) { + test.addInitializer(cmd -> cmd.addArgument("--win-menu")); + } + + if (withDesktopShortcut) { + test.addInitializer(cmd -> cmd.addArgument("--win-shortcut")); + } + + test.run(); + } + + private void setPackageName(JPackageCommand cmd) { + StringBuilder sb = new StringBuilder(cmd.name()); + sb.append("With"); + if (withShortcutPrompt) { + sb.append("ShortcutPrompt"); + } + if (withStartMenuShortcut) { + sb.append("StartMenu"); + } + if (withDesktopShortcut) { + sb.append("Desktop"); + } + cmd.setArgumentValue("--name", sb.toString()); + } + + private final boolean withStartMenuShortcut; + private final boolean withDesktopShortcut; + private final boolean withShortcutPrompt; +}