Tod*_*ebb 7 java windows smb jfilechooser network-share
我们软件的用户需要在我们的Java swing应用程序中浏览Windows 10上的网络共享,但是swing的JFileChooser默认情况下不具备此功能.
在这篇相关文章中 如何在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." );
}
Run Code Online (Sandbox Code Playgroud)
但是,我们正在转向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() ) );
}
Run Code Online (Sandbox Code Playgroud)
这很好,但遗憾的是,两个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也是如此
Run Code Online (Sandbox Code Playgroud)'--add-exports', 'java.desktop/sun.awt.shell=ALL-UNNAMED'
它没有用,因为之前的解决方案现在在Java 11(9+)下抛出相同的Exception.
我们已经看到在计算器上的另一个解决方案是使用SmbFile类JCIFS,但由于安全限制是很难供我们使用第三方代码.特别是如果没有使用私有API更新Java 11 JPMS.
有趣的是,JavaFX中的DirectoryChooser没有这个问题.如果用户手动键入网络主机,它将很乐意显示该主机的所有共享名称.如果我们必须这样做,我们将采用这种方式,但是在一个摇摆应用程序的FX阶段之间处理模态是丑陋的,可能是非常多的工作.
仍然希望更简单的解决方法让JFileChooser在Java 11(Java 9+)中显示网络共享!也许有人知道FX DirectoryChooser正在使用的技巧,它可以应用于JFileChooser?
仍在寻找更好的解决方案,但与此同时,由于网络共享的原因,我们决定在 Java 11 中将 JFileChooser 替换为 JavaFX 的 DirectoryChooser。DirectoryChooser 非常乐意允许用户输入根网络共享,它将显示所有网络共享名称。启动 DirectoryChooser 的 UI 仍然是 swing UI,但我们没有遇到模式或焦点问题,至少在 Windows 10 下是这样。
然而,除了必须仔细管理 JavaFX Platform.runLater() 和 SwingUtilities.invokeLater() 之间的线程之外,我们还有一些重大缺点
1)DirectoryChooser不允许多选。JFileChooser 可以。
2) DirectoryChooser 仅允许选择目录而不是文件。 耸耸肩谁需要允许选择目录或文件?愚蠢的。这正是我们的客户在不止一种情况下需要做的。JFileChooser 支持这一点。
3) DirectoryChooser 不允许输入无效路径。啊?是的,事实上,我们的一位客户有一个特定要求,要求输入尚不存在(预配置)的路径,但一旦任务驱动器插入网络就会出现。JFileChooser 允许这样做,并且可以非常方便地导航到您需要的根目录,然后只需键入路径的最后部分(目录或共享名)。
4)DirectoryChooser与应用风格不匹配。在 JavaFX 中,您可以在场景的根节点上设置自己的 CSS,但不能在 DirectoryChooser 上设置。看起来 DirectoryChooser 实际上正在调用本机文件选择器,它应该使用系统配色方案。不幸的是,对于我们在夜间使用“深色模式”操作的飞行员来说,Windows 10 文件资源管理器直到版本 1809 才尊重“深色模式”设置,而我们的客户几乎没有一个拥有该设置。
| 归档时间: |
|
| 查看次数: |
172 次 |
| 最近记录: |