0

我想从我的 Java-App 中的 MS-Word 模板打开一个新文档,但只能设法编辑模板本身。

这是我的情况:在我的 Jar 文件中有一个 word 模板,它被复制到用户指定的位置,因此他/她可以对其进行编辑。之后,应用程序可以打开这个编辑过的模板,将数据插入其中并在word中打开它。这一切都很好(使用 Apache-POI),但最后一步并不完全是我想要的。

通常,当双击一个 word 模板时,Word 会打开一个尚未保存在任何地方的新文档(标题为 Document1)。在我的例子中,Word 打开 word-template 进行编辑(标题为 blablaMyTemplate),这意味着应该从中创建文档的已保存模板。如何使用 Java 从模板打开新创建的文档?

这是我的代码(try/catch 和流关闭省略):

    File bbb = new File(new File(getClass().getProtectionDomain().getCodeSource().getLocation().toURI().getPath()).getParentFile().getParentFile().getAbsolutePath() + "/blablaMyTemplate.dotx");
    if (!bbb.exists()) { //copy file to outside of jar for user editing
        Files.copy(Buchungsbegleitblatt.class.getResourceAsStream("bbb.dotx"), bbb.toPath(), StandardCopyOption.REPLACE_EXISTING);
    }
    File tmp = File.createTempFile("bbb", ".dotx"); //create tmp file to insert data
    InputStream in = new FileInputStream(bbb);
    OutputStream out = new FileOutputStream(tmp);
    XWPFDocument document = new XWPFDocument(in);
    //here, some data is filled into the document using Apache-POI (omitted, because it works fine)
    document.write(out);
    if (Desktop.isDesktopSupported()) {
        Desktop.getDesktop().open(tmp); //this opens the template for editing, it does not create a new doc from template
    }

问题在于最后一行,但我不知道我还能在这里调用什么。

为了让它更清楚一点,这是我在模板文件上获得的上下文菜单的图像以及应该发生的事情:

模板上的上下文菜单

4

2 回答 2

2

你已经准确地描述了这个问题。Desktop.open将完全按照它所说的去做。它将open为分配给文件类型的被调用应用程序执行事件。

您需要的是执行该new事件。这可以通过Word使用启动命令行开关来启动 Word来实现。

在链接的知识库条目中,您可以找到:

...

/ttemplate_name Starts Word with a new document based on a template other than the Normal template.

...

这样做可以使用或Java使用。对于两者,我建议首先将命令解释器作为 shell 启动,然后使用其中的命令来启动应用程序。因此,我们无需知道应用程序的确切路径。Runtime.getRuntime().execProcessBuilderCMDstart

例子:

import java.io.*;

class RuntimeExec {
  public static void main(String[] args) {
      try {
        // Execute a command as a single line
        File f = new File("C:/Users/axel/Documents/The Template.dotx");
        System.out.println(f.getAbsolutePath());
        String cmd = "cmd /C start winword.exe /t\"" + f.getAbsolutePath() + "\"";
        Process child = Runtime.getRuntime().exec(cmd);

      } catch (IOException e) {
        e.printStackTrace();
      }

  }
}

class UseProcessBuilder {
  public static void main(String[] args) {
      try {
        //use ProcessBuilder to have more control
        File f = new File("C:/Users/axel/Documents/The Template.dotx");
        System.out.println(f.getAbsolutePath());
        String application = "winword.exe";
        String switchNewFromTemplate = "/t";
        String file = f.getAbsolutePath(); 
        ProcessBuilder pb = new ProcessBuilder("cmd", "/C", "start", application, switchNewFromTemplate+file);
        Process process = pb.start();

      } catch (IOException e) {
        e.printStackTrace();
      }
  }
}

有一种可能是不显式启动winword应用程序。如果我们将空字符串作为应用程序名称,该start命令具有根据给定文件的文件扩展名执行默认操作的功能:""

start "" "The name of the file.ext"

例子:

start "" "The name of the file.dotx"

new这将在winword应用程序中执行与dotx注册表数据库中的扩展相关的默认操作。

所以:

class RuntimeExec {
  public static void main(String[] args) {
      try {
        // Execute a command as a single line
        File f = new File("C:/Users/Axel Richter/Documents/The Template.dotx");
        System.out.println(f.getAbsolutePath());
        String cmd = "cmd /C start \"\" \"" + f.getAbsolutePath() + "\"";
        Process child = Runtime.getRuntime().exec(cmd);

        InputStream in = child.getErrorStream();
        int c;
        while ((c = in.read()) != -1) {
            System.out.print((char)c);
        }
        in.close();

      } catch (IOException e) {
        e.printStackTrace();
      }

  }
}

class UseProcessBuilder {
  public static void main(String[] args) {
      try {
        //use ProcessBuilder to have more control
        File f = new File("C:/Users/Axel Richter/Documents/The Template.dotx");
        System.out.println(f.getAbsolutePath());
        String file = f.getAbsolutePath(); 
        ProcessBuilder pb = new ProcessBuilder("cmd", "/C", "start", "\"\"", file);
        Process process = pb.start();

        InputStream in = process.getErrorStream();
        int c;
        while ((c = in.read()) != -1) {
            System.out.print((char)c);
        }
        in.close();

      } catch (IOException e) {
        e.printStackTrace();
      }
  }
}
于 2016-10-01T06:20:15.320 回答
0

一种方法是在模板上启动一个进程,以便 Windows 处理打开并使用默认意图。自从我接触 Java 以来已经有一段时间了,但如果它像 C# 一样,它将类似于new Process(tmp).Start().

不过,我不确定这是否是您正在寻找的东西。

于 2016-09-30T15:26:17.243 回答