如何在Java 9+中使用JFileChooser显示网络共享?

问题描述 投票:6回答:1

我们软件的用户需要在我们的Java swing应用程序中浏览Windows 10上的网络共享,但是swing的JFileChooser默认情况下不具备此功能。

在这个相关的帖子中How to navigate to a network host in JFileChooser?使用ShellFolder(sun私有API)来设置一个体面的解决方案来设置JFileChooser的当前目录,并且我们在过去几年中一直使用这种方法进行一些修改而没有任何问题。

public static File getNetworkShareFolder( final File folder ) throws IllegalArgumentException {
  final File file = new NonCanonicalizingFile( folder.getPath() );
  if( isNetworkShareFolder( file ) ) { // assume Win32ShellFolderManager2 will be present
     try {
        // JRE-13272 -PRIVATE API that must eventually be swapped for Java 9 alternative
        // Using reflection because Win32ShellFolderManager2 may not exist in rt.jar on Linux build server
        final Class win32ShellMgr = Class.forName( "sun.awt.shell.Win32ShellFolderManager2" );
        // get static creation method from class, execute it
        final Method cfMethod = win32ShellMgr.getMethod( "createShellFolder", File.class );
        return (ShellFolder) cfMethod.invoke( win32ShellMgr.newInstance(), file );
     } catch( final Exception xx ) {
        xx.printStackTrace();
     }
  }
  throw new IllegalArgumentException( "Given path is not a Windows network share folder." ); 
} 

但是,我们正在转向Java 11,并且在Java 9中,私有API被封装,我们已经被要求不再使用它们。不用担心,OpenJDK中的替换API已进入FileSystemView,在swing filechooser的子包中。

sun.awt.shell.ShellFolder.isComputerNode(File) - > javax.swing.filechooser.FileSystemView.getFileSystemView()。isComputerNode(File)sun.awt.shell.ShellFolder.getShellFolder(File) - > javax.swing.filechooser。 FileSystemView.getFileSystemView()。getLinkLocation(File)

所以现在的代码变成了

public static File getNetworkShareFolder( final File folder ) throws IllegalArgumentException {
   final File file = new NonCanonicalizingFile( folder.getPath() );
   if( isNetworkShareFolder( file ) ) { 
      try {
         return FileSystemView.getFileSystemView().getLinkLocation( file );
      } catch( final Exception xx ) {
         xx.printStackTrace();
      }
   }
   throw new IllegalArgumentException( "Given path is not a Windows network share folder." );
}

public static boolean isNetworkShareFolder( final File folder ) {
   return FileSystemView.getFileSystemView().isComputerNode( new NonCanonicalizingFile( folder.getPath() ) );
}

这将是很好的,但不幸的是,两个getShellFolder()和getLinkLocation()在Java 11下抛出了一个在Java 8下没有抛出的异常。

java.nio.file.InvalidPathException:UNC路径在java.base / sun.nio.fs上的java.base / sun.nio.fs.WindowsPathParser.parse(WindowsPathParser.java:118)中缺少sharename:\ 100.212.51.37。 WindowsPathParser.parse(WindowsPathParser.java:77)位于java.base / sun.nio.fs.WindowsPath.parse(WindowsPath.java:92)java.base / sun.nio.fs.WindowsFileSystem.getPath(WindowsFileSystem.java: 229)java.desktop / sun的java.base / java.nio.file.Path.of(Path.java:147)java.base / java.nio.file.Paths.get(Paths.java:69) .awt.shell.ShellFolder.getShellFolder(ShellFolder.java:247)at java.desktop / javax.swing.filechooser.FileSystemView.getLinkLocation(FileSystemView.java:641)

看来它现在认为UNC路径无效,除非它有实际的共享名称,即“\\ 100.212.51.37”无效,但“\\ 100.212.51.37 \ myShare”没问题。

现在,如果您获得UNC路径“\\ 100.212.51.37 \ myShare”的shell文件夹,然后获取getParent(),则首先获得我们想要的“\\ 100.212.51.37 \”的shell文件夹。不幸的是,这种解决方法对于我们的客户来说是不可行的,因为鸡和鸡蛋问题 - 用户通常还不知道任何实际的共享名称,这是他们想要首先浏览的内容!

Argh - 这在Java 8下运行良好,但在Java 11中,即使你打破封装使用原来的ShellFolder私有API也是如此

'--add-exports', 'java.desktop/sun.awt.shell=ALL-UNNAMED'

它没有用,因为之前的解决方案现在在Java 11(9+)下抛出相同的Exception。

我们在StackOverflow上看到的另一个解决方案是在JCIFS中使用SmbFile类,但由于安全限制,我们很难使用第三方代码。特别是如果没有使用私有API更新Java 11 JPMS。

有趣的是,JavaFX中的DirectoryChooser没有这个问题。如果用户手动键入网络主机,它将很乐意显示该主机的所有共享名称。如果我们必须这样做,我们将采用这种方式,但是在一个摇摆应用程序的FX阶段之间处理模态是丑陋的,并且可能是非常多的工作。

仍然希望更简单的解决方法让JFileChooser在Java 11(Java 9+)中显示网络共享!也许有人知道FX DirectoryChooser正在使用的技巧,它可以应用于JFileChooser?

java windows jfilechooser smb network-share
1个回答
0
投票

仍在寻找更好的解决方案,但与此同时,由于网络共享,我们决定用Java 11中的JavaFX的DirectoryChooser替换JFileChooser。 DirectoryChooser非常乐意允许用户键入根网络共享,它将显示所有网络共享名称。启动DirectoryChooser的UI仍然是一个swing UI,但我们没有遇到模态或焦点问题,至少在Windows 10下。

但是,除了必须仔细管理JavaFX Platform.runLater()和SwingUtilities.invokeLater()之间的线程之外,还有一些重大缺陷。

1)DirectoryChooser不允许多项选择。 JFileChooser。

2)DirectoryChooser ONLY允许选择目录而不是文件。 Shrugs谁需要允许选择目录或文件?愚蠢。只有这就是我们的客户需要在多个实例中完成的工作。 JFileChooser支持这一点。

3)DirectoryChooser不允许输入无效路径。咦?是的,事实上,我们的一个客户有一个特定的要求,即输入一个尚不存在的路径(预先配置),但是一旦任务驱动器插入网络就会有。 JFileChooser允许这样做,并且可以非常方便地导航到您需要的根目录,然后只需键入路径的最后一部分(目录或共享名称)。

4)DirectoryChooser与应用程序样式不匹配。在JavaFX中,您可以在场景的根节点上设置自己的CSS,但不能在DirectoryChooser上设置。看起来DirectoryChooser实际上正在调用本机文件选择器,它应该使用系统颜色方案。不幸的是,我们的飞行员在晚上使用“暗模式”操作,Windows 10文件资源管理器不会尊重“暗模式”设置,直到版本1809,几乎没有我们的客户拥有。

© www.soinside.com 2019 - 2024. All rights reserved.