1 /* 2 * Copyright (c) 1998, 2016, Oracle and/or its affiliates. All rights reserved. 3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. 4 * 5 * This code is free software; you can redistribute it and/or modify it 6 * under the terms of the GNU General Public License version 2 only, as 7 * published by the Free Software Foundation. Oracle designates this 8 * particular file as subject to the "Classpath" exception as provided 9 * by Oracle in the LICENSE file that accompanied this code. 10 * 11 * This code is distributed in the hope that it will be useful, but WITHOUT 12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 14 * version 2 for more details (a copy is included in the LICENSE file that 15 * accompanied this code). 16 * 17 * You should have received a copy of the GNU General Public License version 18 * 2 along with this work; if not, write to the Free Software Foundation, 19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. 20 * 21 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA 22 * or visit www.oracle.com if you need additional information or have any 23 * questions. 24 */ 25 26 package javax.swing.filechooser; 27 28 29 import javax.swing.*; 30 31 import java.awt.Image; 32 import java.beans.BeanProperty; 33 import java.beans.JavaBean; 34 import java.io.File; 35 import java.io.FileNotFoundException; 36 import java.io.IOException; 37 import java.text.MessageFormat; 38 import java.util.List; 39 import java.util.ArrayList; 40 import java.lang.ref.WeakReference; 41 import java.beans.PropertyChangeListener; 42 import java.beans.PropertyChangeEvent; 43 import java.beans.PropertyChangeSupport; 44 import java.security.AccessController; 45 import java.security.PrivilegedAction; 46 import java.util.Arrays; 47 import java.util.Collections; 48 import java.util.Objects; 49 import sun.awt.OSInfo; 50 51 import sun.awt.shell.*; 52 53 /** 54 * FileSystemView is JFileChooser's gateway to the 55 * file system. Since the JDK1.1 File API doesn't allow 56 * access to such information as root partitions, file type 57 * information, or hidden file bits, this class is designed 58 * to intuit as much OS-specific file system information as 59 * possible. 60 * 61 * <p> 62 * 63 * Java Licensees may want to provide a different implementation of 64 * FileSystemView to better handle a given operating system. 65 * 66 * @author Jeff Dinkins 67 */ 68 69 // PENDING(jeff) - need to provide a specification for 70 // how Mac/OS2/BeOS/etc file systems can modify FileSystemView 71 // to handle their particular type of file system. 72 73 public abstract class FileSystemView { 74 75 static FileSystemView windowsFileSystemView = null; 76 static FileSystemView unixFileSystemView = null; 77 //static FileSystemView macFileSystemView = null; 78 static FileSystemView genericFileSystemView = null; 79 80 private boolean useSystemExtensionHiding = 81 UIManager.getDefaults().getBoolean("FileChooser.useSystemExtensionHiding"); 82 83 /** 84 * Returns the file system view. 85 * @return the file system view 86 */ 87 public static FileSystemView getFileSystemView() { 88 if(File.separatorChar == '\\') { 89 if(windowsFileSystemView == null) { 90 windowsFileSystemView = new WindowsFileSystemView(); 91 } 92 return windowsFileSystemView; 93 } 94 95 if(File.separatorChar == '/') { 96 if(unixFileSystemView == null) { 97 unixFileSystemView = new UnixFileSystemView(); 98 } 99 return unixFileSystemView; 100 } 101 102 // if(File.separatorChar == ':') { 103 // if(macFileSystemView == null) { 104 // macFileSystemView = new MacFileSystemView(); 105 // } 106 // return macFileSystemView; 107 //} 108 109 if(genericFileSystemView == null) { 110 genericFileSystemView = new GenericFileSystemView(); 111 } 112 return genericFileSystemView; 113 } 114 115 /** 116 * Constructs a FileSystemView. 117 */ 118 public FileSystemView() { 119 final WeakReference<FileSystemView> weakReference = new WeakReference<FileSystemView>(this); 120 121 UIManager.addPropertyChangeListener(new PropertyChangeListener() { 122 public void propertyChange(PropertyChangeEvent evt) { 123 FileSystemView fileSystemView = weakReference.get(); 124 125 if (fileSystemView == null) { 126 // FileSystemView was destroyed 127 UIManager.removePropertyChangeListener(this); 128 } else { 129 if (evt.getPropertyName().equals("lookAndFeel")) { 130 fileSystemView.useSystemExtensionHiding = 131 UIManager.getDefaults().getBoolean("FileChooser.useSystemExtensionHiding"); 132 } 133 } 134 } 135 }); 136 } 137 138 /** 139 * Determines if the given file is a root in the navigable tree(s). 140 * Examples: Windows 98 has one root, the Desktop folder. DOS has one root 141 * per drive letter, <code>C:\</code>, <code>D:\</code>, etc. Unix has one root, 142 * the <code>"/"</code> directory. 143 * 144 * The default implementation gets information from the <code>ShellFolder</code> class. 145 * 146 * @param f a <code>File</code> object representing a directory 147 * @return <code>true</code> if <code>f</code> is a root in the navigable tree. 148 * @see #isFileSystemRoot 149 */ 150 public boolean isRoot(File f) { 151 if (f == null || !f.isAbsolute()) { 152 return false; 153 } 154 155 File[] roots = getRoots(); 156 for (File root : roots) { 157 if (root.equals(f)) { 158 return true; 159 } 160 } 161 return false; 162 } 163 164 /** 165 * Returns true if the file (directory) can be visited. 166 * Returns false if the directory cannot be traversed. 167 * 168 * @param f the <code>File</code> 169 * @return <code>true</code> if the file/directory can be traversed, otherwise <code>false</code> 170 * @see JFileChooser#isTraversable 171 * @see FileView#isTraversable 172 * @since 1.4 173 */ 174 public Boolean isTraversable(File f) { 175 return Boolean.valueOf(f.isDirectory()); 176 } 177 178 /** 179 * Name of a file, directory, or folder as it would be displayed in 180 * a system file browser. Example from Windows: the "M:\" directory 181 * displays as "CD-ROM (M:)" 182 * 183 * The default implementation gets information from the ShellFolder class. 184 * 185 * @param f a <code>File</code> object 186 * @return the file name as it would be displayed by a native file chooser 187 * @see JFileChooser#getName 188 * @since 1.4 189 */ 190 public String getSystemDisplayName(File f) { 191 if (f == null) { 192 return null; 193 } 194 195 String name = f.getName(); 196 197 if (!name.equals("..") && !name.equals(".") && 198 (useSystemExtensionHiding || !isFileSystem(f) || isFileSystemRoot(f)) && 199 (f instanceof ShellFolder || f.exists())) { 200 201 try { 202 name = getShellFolder(f).getDisplayName(); 203 } catch (FileNotFoundException e) { 204 return null; 205 } 206 207 if (name == null || name.length() == 0) { 208 name = f.getPath(); // e.g. "/" 209 } 210 } 211 212 return name; 213 } 214 215 /** 216 * Type description for a file, directory, or folder as it would be displayed in 217 * a system file browser. Example from Windows: the "Desktop" folder 218 * is described as "Desktop". 219 * 220 * Override for platforms with native ShellFolder implementations. 221 * 222 * @param f a <code>File</code> object 223 * @return the file type description as it would be displayed by a native file chooser 224 * or null if no native information is available. 225 * @see JFileChooser#getTypeDescription 226 * @since 1.4 227 */ 228 public String getSystemTypeDescription(File f) { 229 return null; 230 } 231 232 /** 233 * Icon for a file, directory, or folder as it would be displayed in 234 * a system file browser. Example from Windows: the "M:\" directory 235 * displays a CD-ROM icon. 236 * 237 * The default implementation gets information from the ShellFolder class. 238 * 239 * @param f a <code>File</code> object 240 * @return an icon as it would be displayed by a native file chooser 241 * @see JFileChooser#getIcon 242 * @since 1.4 243 */ 244 public Icon getSystemIcon(File f) { 245 if (f == null) { 246 return null; 247 } 248 249 ShellFolder sf; 250 251 try { 252 sf = getShellFolder(f); 253 } catch (FileNotFoundException e) { 254 return null; 255 } 256 257 Image img = sf.getIcon(false); 258 259 if (img != null) { 260 return new ImageIcon(img, sf.getFolderType()); 261 } else { 262 return UIManager.getIcon(f.isDirectory() ? "FileView.directoryIcon" : "FileView.fileIcon"); 263 } 264 } 265 266 /** 267 * On Windows, a file can appear in multiple folders, other than its 268 * parent directory in the filesystem. Folder could for example be the 269 * "Desktop" folder which is not the same as file.getParentFile(). 270 * 271 * @param folder a <code>File</code> object representing a directory or special folder 272 * @param file a <code>File</code> object 273 * @return <code>true</code> if <code>folder</code> is a directory or special folder and contains <code>file</code>. 274 * @since 1.4 275 */ 276 public boolean isParent(File folder, File file) { 277 if (folder == null || file == null) { 278 return false; 279 } else if (folder instanceof ShellFolder) { 280 File parent = file.getParentFile(); 281 if (parent != null && parent.equals(folder)) { 282 return true; 283 } 284 File[] children = getFiles(folder, false); 285 for (File child : children) { 286 if (file.equals(child)) { 287 return true; 288 } 289 } 290 return false; 291 } else { 292 return folder.equals(file.getParentFile()); 293 } 294 } 295 296 /** 297 * 298 * @param parent a <code>File</code> object representing a directory or special folder 299 * @param fileName a name of a file or folder which exists in <code>parent</code> 300 * @return a File object. This is normally constructed with <code>new 301 * File(parent, fileName)</code> except when parent and child are both 302 * special folders, in which case the <code>File</code> is a wrapper containing 303 * a <code>ShellFolder</code> object. 304 * @since 1.4 305 */ 306 public File getChild(File parent, String fileName) { 307 if (parent instanceof ShellFolder) { 308 File[] children = getFiles(parent, false); 309 for (File child : children) { 310 if (child.getName().equals(fileName)) { 311 return child; 312 } 313 } 314 } 315 return createFileObject(parent, fileName); 316 } 317 318 319 /** 320 * Checks if <code>f</code> represents a real directory or file as opposed to a 321 * special folder such as <code>"Desktop"</code>. Used by UI classes to decide if 322 * a folder is selectable when doing directory choosing. 323 * 324 * @param f a <code>File</code> object 325 * @return <code>true</code> if <code>f</code> is a real file or directory. 326 * @since 1.4 327 */ 328 public boolean isFileSystem(File f) { 329 if (f instanceof ShellFolder) { 330 ShellFolder sf = (ShellFolder)f; 331 // Shortcuts to directories are treated as not being file system objects, 332 // so that they are never returned by JFileChooser. 333 return sf.isFileSystem() && !(sf.isLink() && sf.isDirectory()); 334 } else { 335 return true; 336 } 337 } 338 339 /** 340 * Creates a new folder with a default folder name. 341 * 342 * @param containingDir a {@code File} object denoting directory to contain the new folder 343 * @return a {@code File} object denoting the newly created folder 344 * @throws IOException if new folder could not be created 345 */ 346 public abstract File createNewFolder(File containingDir) throws IOException; 347 348 /** 349 * Returns whether a file is hidden or not. 350 * 351 * @param f a {@code File} object 352 * @return true if the given {@code File} denotes a hidden file 353 */ 354 public boolean isHiddenFile(File f) { 355 return f.isHidden(); 356 } 357 358 359 /** 360 * Is dir the root of a tree in the file system, such as a drive 361 * or partition. Example: Returns true for "C:\" on Windows 98. 362 * 363 * @param dir a <code>File</code> object representing a directory 364 * @return <code>true</code> if <code>f</code> is a root of a filesystem 365 * @see #isRoot 366 * @since 1.4 367 */ 368 public boolean isFileSystemRoot(File dir) { 369 return ShellFolder.isFileSystemRoot(dir); 370 } 371 372 /** 373 * Used by UI classes to decide whether to display a special icon 374 * for drives or partitions, e.g. a "hard disk" icon. 375 * 376 * The default implementation has no way of knowing, so always returns false. 377 * 378 * @param dir a directory 379 * @return <code>false</code> always 380 * @since 1.4 381 */ 382 public boolean isDrive(File dir) { 383 return false; 384 } 385 386 /** 387 * Used by UI classes to decide whether to display a special icon 388 * for a floppy disk. Implies isDrive(dir). 389 * 390 * The default implementation has no way of knowing, so always returns false. 391 * 392 * @param dir a directory 393 * @return <code>false</code> always 394 * @since 1.4 395 */ 396 public boolean isFloppyDrive(File dir) { 397 return false; 398 } 399 400 /** 401 * Used by UI classes to decide whether to display a special icon 402 * for a computer node, e.g. "My Computer" or a network server. 403 * 404 * The default implementation has no way of knowing, so always returns false. 405 * 406 * @param dir a directory 407 * @return <code>false</code> always 408 * @since 1.4 409 */ 410 public boolean isComputerNode(File dir) { 411 return ShellFolder.isComputerNode(dir); 412 } 413 414 415 /** 416 * Returns all root partitions on this system. For example, on 417 * Windows, this would be the "Desktop" folder, while on DOS this 418 * would be the A: through Z: drives. 419 * 420 * @return an array of {@code File} objects representing all root partitions 421 * on this system 422 */ 423 public File[] getRoots() { 424 // Don't cache this array, because filesystem might change 425 File[] roots = (File[])ShellFolder.get("roots"); 426 427 for (int i = 0; i < roots.length; i++) { 428 if (isFileSystemRoot(roots[i])) { 429 roots[i] = createFileSystemRoot(roots[i]); 430 } 431 } 432 return roots; 433 } 434 435 436 // Providing default implementations for the remaining methods 437 // because most OS file systems will likely be able to use this 438 // code. If a given OS can't, override these methods in its 439 // implementation. 440 441 /** 442 * Returns the home directory. 443 * @return the home directory 444 */ 445 public File getHomeDirectory() { 446 return createFileObject(System.getProperty("user.home")); 447 } 448 449 /** 450 * Return the user's default starting directory for the file chooser. 451 * 452 * @return a <code>File</code> object representing the default 453 * starting folder 454 * @since 1.4 455 */ 456 public File getDefaultDirectory() { 457 File f = (File)ShellFolder.get("fileChooserDefaultFolder"); 458 if (isFileSystemRoot(f)) { 459 f = createFileSystemRoot(f); 460 } 461 return f; 462 } 463 464 /** 465 * Returns a File object constructed in dir from the given filename. 466 * 467 * @param dir an abstract pathname denoting a directory 468 * @param filename a {@code String} representation of a pathname 469 * @return a {@code File} object created from {@code dir} and {@code filename} 470 */ 471 public File createFileObject(File dir, String filename) { 472 if(dir == null) { 473 return new File(filename); 474 } else { 475 return new File(dir, filename); 476 } 477 } 478 479 /** 480 * Returns a File object constructed from the given path string. 481 * 482 * @param path {@code String} representation of path 483 * @return a {@code File} object created from the given {@code path} 484 */ 485 public File createFileObject(String path) { 486 File f = new File(path); 487 if (isFileSystemRoot(f)) { 488 f = createFileSystemRoot(f); 489 } 490 return f; 491 } 492 493 494 /** 495 * Gets the list of shown (i.e. not hidden) files. 496 * 497 * @param dir the root directory of files to be returned 498 * @param useFileHiding determine if hidden files are returned 499 * @return an array of {@code File} objects representing files and 500 * directories in the given {@code dir}. It includes hidden 501 * files if {@code useFileHiding} is false. 502 */ 503 public File[] getFiles(File dir, boolean useFileHiding) { 504 List<File> files = new ArrayList<File>(); 505 506 // add all files in dir 507 if (!(dir instanceof ShellFolder)) { 508 try { 509 dir = getShellFolder(dir); 510 } catch (FileNotFoundException e) { 511 return new File[0]; 512 } 513 } 514 515 File[] names = ((ShellFolder) dir).listFiles(!useFileHiding); 516 517 if (names == null) { 518 return new File[0]; 519 } 520 521 for (File f : names) { 522 if (Thread.currentThread().isInterrupted()) { 523 break; 524 } 525 526 if (!(f instanceof ShellFolder)) { 527 if (isFileSystemRoot(f)) { 528 f = createFileSystemRoot(f); 529 } 530 try { 531 f = ShellFolder.getShellFolder(f); 532 } catch (FileNotFoundException e) { 533 // Not a valid file (wouldn't show in native file chooser) 534 // Example: C:\pagefile.sys 535 continue; 536 } catch (InternalError e) { 537 // Not a valid file (wouldn't show in native file chooser) 538 // Example C:\Winnt\Profiles\joe\history\History.IE5 539 continue; 540 } 541 } 542 if (!useFileHiding || !isHiddenFile(f)) { 543 files.add(f); 544 } 545 } 546 547 return files.toArray(new File[files.size()]); 548 } 549 550 551 552 /** 553 * Returns the parent directory of <code>dir</code>. 554 * @param dir the <code>File</code> being queried 555 * @return the parent directory of <code>dir</code>, or 556 * <code>null</code> if <code>dir</code> is <code>null</code> 557 */ 558 public File getParentDirectory(File dir) { 559 if (dir == null || !dir.exists()) { 560 return null; 561 } 562 563 ShellFolder sf; 564 565 try { 566 sf = getShellFolder(dir); 567 } catch (FileNotFoundException e) { 568 return null; 569 } 570 571 File psf = sf.getParentFile(); 572 573 if (psf == null) { 574 return null; 575 } 576 577 if (isFileSystem(psf)) { 578 File f = psf; 579 if (!f.exists()) { 580 // This could be a node under "Network Neighborhood". 581 File ppsf = psf.getParentFile(); 582 if (ppsf == null || !isFileSystem(ppsf)) { 583 // We're mostly after the exists() override for windows below. 584 f = createFileSystemRoot(f); 585 } 586 } 587 return f; 588 } else { 589 return psf; 590 } 591 } 592 593 /** 594 * Returns an array of files representing the values to show by default in 595 * the file chooser selector. 596 * 597 * @return an array of {@code File} objects. 598 * @throws SecurityException if the caller does not have necessary 599 * permissions 600 * @since 9 601 */ 602 public File[] getChooserComboBoxFiles() { 603 return (File[]) ShellFolder.get("fileChooserComboBoxFolders"); 604 } 605 606 /** 607 * Returns whether the specified file denotes a shell interpreted link which 608 * can be obtained by the {@link #getLinkLocation(File)}. 609 * 610 * @param file a file 611 * @return whether this is a link 612 * @throws NullPointerException if {@code file} equals {@code null} 613 * @throws SecurityException if the caller does not have necessary 614 * permissions 615 * @see #getLinkLocation(File) 616 * @since 9 617 */ 618 public boolean isLink(File file) { 619 if (file == null) { 620 throw new NullPointerException("file is null"); 621 } 622 try { 623 return ShellFolder.getShellFolder(file).isLink(); 624 } catch (FileNotFoundException e) { 625 return false; 626 } 627 } 628 629 /** 630 * Returns the regular file referenced by the specified link file if 631 * the specified file is a shell interpreted link. 632 * Returns {@code null} if the specified file is not 633 * a shell interpreted link. 634 * 635 * @param file a file 636 * @return the linked file or {@code null}. 637 * @throws FileNotFoundException if the linked file does not exist 638 * @throws NullPointerException if {@code file} equals {@code null} 639 * @throws SecurityException if the caller does not have necessary 640 * permissions 641 * @since 9 642 */ 643 public File getLinkLocation(File file) throws FileNotFoundException { 644 if (file == null) { 645 throw new NullPointerException("file is null"); 646 } 647 ShellFolder shellFolder; 648 try { 649 shellFolder = ShellFolder.getShellFolder(file); 650 } catch (FileNotFoundException e) { 651 return null; 652 } 653 return shellFolder.isLink() ? shellFolder.getLinkLocation() : null; 654 } 655 656 /** 657 * A group of essential folder that are intented to appear in a left pane of 658 * a file chooser. 659 */ 660 @JavaBean(description = "A group of essential folder that are intented to " 661 + "appear in a left pane of a file chooser. ") 662 public static class EssentialFolderSection { 663 664 public static final String FOLDERS_PROPERTY = "folders"; 665 666 private final PropertyChangeSupport propertyChangeSupport = 667 new PropertyChangeSupport(this); 668 669 private List<File> folders = Collections.emptyList(); 670 671 /** 672 * Creates a new {@code EssentialFolderSection} with no folders. 673 */ 674 public EssentialFolderSection() { 675 } 676 677 /** 678 * Creates a new {@code EssentialFolderSection} with the specified 679 * folders. 680 * 681 * @param folders 682 */ 683 public EssentialFolderSection(File... folders) { 684 setFolders(Arrays.asList(folders)); 685 } 686 687 688 /** 689 * Gets a list of the essential folders in this section. 690 * 691 * @return the value of folders 692 */ 693 @SuppressWarnings("ReturnOfCollectionOrArrayField") // already an unmodifiableList or emptyList 694 public List<File> getFolders() { 695 return folders; 696 } 697 698 /** 699 * Updates the list of the essential folders in this section to the specified list. 700 * 701 * @param folders 702 * @throws NullPointerException if the argument is null 703 */ 704 @BeanProperty(description = "The folders that are in ", bound = true, visualUpdate =true) 705 public void setFolders(List<File> folders) { 706 Objects.requireNonNull(folders); 707 folders = Collections.unmodifiableList(new ArrayList<>(folders)); 708 709 List<File> oldFolders = this.folders; 710 this.folders = folders; 711 propertyChangeSupport.firePropertyChange(FOLDERS_PROPERTY, oldFolders, folders); 712 } 713 714 /** 715 * Adds a PropertyChangeListener to the listener list. Currently the 716 * listener will be fired only when the 717 * {@link #setFolders(java.util.List) folders} property is changed, 718 * but in the future more properties will be added. 719 * @param listener the PropertyChangeListener to be added 720 */ 721 public void addPropertyChangeListener(PropertyChangeListener listener) { 722 propertyChangeSupport.addPropertyChangeListener(listener); 723 } 724 725 /** 726 * Removes a PropertyChangeListener from the listener list. If listener is null, 727 * or was never added, no exception is thrown and no action is taken. 728 * 729 * @param listener the PropertyChangeListener to be removed (registered using 730 * {@link #addPropertyChangeListener(java.beans.PropertyChangeListener)}) 731 */ 732 public void removePropertyChangeListener(PropertyChangeListener listener) { 733 propertyChangeSupport.removePropertyChangeListener(listener); 734 } 735 736 } 737 738 public EssentialFolderSection[] getEssentialFolders() { 739 return FileSystemView.getFileSystemView().getEssentialFolders(); 740 } 741 742 /** 743 * Throws {@code FileNotFoundException} if file not found or current thread was interrupted 744 */ 745 ShellFolder getShellFolder(File f) throws FileNotFoundException { 746 if (!(f instanceof ShellFolder) && !(f instanceof FileSystemRoot) && isFileSystemRoot(f)) { 747 f = createFileSystemRoot(f); 748 } 749 750 try { 751 return ShellFolder.getShellFolder(f); 752 } catch (InternalError e) { 753 System.err.println("FileSystemView.getShellFolder: f="+f); 754 e.printStackTrace(); 755 return null; 756 } 757 } 758 759 /** 760 * Creates a new <code>File</code> object for <code>f</code> with correct 761 * behavior for a file system root directory. 762 * 763 * @param f a <code>File</code> object representing a file system root 764 * directory, for example "/" on Unix or "C:\" on Windows. 765 * @return a new <code>File</code> object 766 * @since 1.4 767 */ 768 protected File createFileSystemRoot(File f) { 769 return new FileSystemRoot(f); 770 } 771 772 @SuppressWarnings("serial") // Same-version serialization only 773 static class FileSystemRoot extends File { 774 public FileSystemRoot(File f) { 775 super(f,""); 776 } 777 778 public FileSystemRoot(String s) { 779 super(s); 780 } 781 782 public boolean isDirectory() { 783 return true; 784 } 785 786 public String getName() { 787 return getPath(); 788 } 789 } 790 } 791 792 /** 793 * FileSystemView that handles some specific unix-isms. 794 */ 795 class UnixFileSystemView extends FileSystemView { 796 797 private static final String newFolderString = 798 UIManager.getString("FileChooser.other.newFolder"); 799 private static final String newFolderNextString = 800 UIManager.getString("FileChooser.other.newFolder.subsequent"); 801 802 /** 803 * Creates a new folder with a default folder name. 804 */ 805 public File createNewFolder(File containingDir) throws IOException { 806 if(containingDir == null) { 807 throw new IOException("Containing directory is null:"); 808 } 809 File newFolder; 810 // Unix - using OpenWindows' default folder name. Can't find one for Motif/CDE. 811 newFolder = createFileObject(containingDir, newFolderString); 812 int i = 1; 813 while (newFolder.exists() && i < 100) { 814 newFolder = createFileObject(containingDir, MessageFormat.format( 815 newFolderNextString, i)); 816 i++; 817 } 818 819 if(newFolder.exists()) { 820 throw new IOException("Directory already exists:" + newFolder.getAbsolutePath()); 821 } else { 822 if(!newFolder.mkdirs()) { 823 throw new IOException(newFolder.getAbsolutePath()); 824 } 825 } 826 827 return newFolder; 828 } 829 830 public boolean isFileSystemRoot(File dir) { 831 return dir != null && dir.getAbsolutePath().equals("/"); 832 } 833 834 public boolean isDrive(File dir) { 835 return isFloppyDrive(dir); 836 } 837 838 public boolean isFloppyDrive(File dir) { 839 // Could be looking at the path for Solaris, but wouldn't be reliable. 840 // For example: 841 // return (dir != null && dir.getAbsolutePath().toLowerCase().startsWith("/floppy")); 842 return false; 843 } 844 845 public boolean isComputerNode(File dir) { 846 if (dir != null) { 847 String parent = dir.getParent(); 848 if (parent != null && parent.equals("/net")) { 849 return true; 850 } 851 } 852 return false; 853 } 854 855 @Override 856 public EssentialFolderSection[] getEssentialFolders() { 857 return new EssentialFolderSection[]{ 858 new EssentialFolderSection(getChooserComboBoxFiles()) 859 }; 860 } 861 } 862 863 864 /** 865 * FileSystemView that handles some specific windows concepts. 866 */ 867 class WindowsFileSystemView extends FileSystemView { 868 869 private static final String newFolderString = 870 UIManager.getString("FileChooser.win32.newFolder"); 871 private static final String newFolderNextString = 872 UIManager.getString("FileChooser.win32.newFolder.subsequent"); 873 874 public Boolean isTraversable(File f) { 875 return Boolean.valueOf(isFileSystemRoot(f) || isComputerNode(f) || f.isDirectory()); 876 } 877 878 public File getChild(File parent, String fileName) { 879 if (fileName.startsWith("\\") 880 && !fileName.startsWith("\\\\") 881 && isFileSystem(parent)) { 882 883 //Path is relative to the root of parent's drive 884 String path = parent.getAbsolutePath(); 885 if (path.length() >= 2 886 && path.charAt(1) == ':' 887 && Character.isLetter(path.charAt(0))) { 888 889 return createFileObject(path.substring(0, 2) + fileName); 890 } 891 } 892 return super.getChild(parent, fileName); 893 } 894 895 /** 896 * Type description for a file, directory, or folder as it would be displayed in 897 * a system file browser. Example from Windows: the "Desktop" folder 898 * is described as "Desktop". 899 * 900 * The Windows implementation gets information from the ShellFolder class. 901 */ 902 public String getSystemTypeDescription(File f) { 903 if (f == null) { 904 return null; 905 } 906 907 try { 908 return getShellFolder(f).getFolderType(); 909 } catch (FileNotFoundException e) { 910 return null; 911 } 912 } 913 914 /** 915 * @return the Desktop folder. 916 */ 917 public File getHomeDirectory() { 918 File[] roots = getRoots(); 919 return (roots.length == 0) ? null : roots[0]; 920 } 921 922 /** 923 * Creates a new folder with a default folder name. 924 */ 925 public File createNewFolder(File containingDir) throws IOException { 926 if(containingDir == null) { 927 throw new IOException("Containing directory is null:"); 928 } 929 // Using NT's default folder name 930 File newFolder = createFileObject(containingDir, newFolderString); 931 int i = 2; 932 while (newFolder.exists() && i < 100) { 933 newFolder = createFileObject(containingDir, MessageFormat.format( 934 newFolderNextString, i)); 935 i++; 936 } 937 938 if(newFolder.exists()) { 939 throw new IOException("Directory already exists:" + newFolder.getAbsolutePath()); 940 } else { 941 if(!newFolder.mkdirs()) { 942 throw new IOException(newFolder.getAbsolutePath()); 943 } 944 } 945 946 return newFolder; 947 } 948 949 public boolean isDrive(File dir) { 950 return isFileSystemRoot(dir); 951 } 952 953 public boolean isFloppyDrive(final File dir) { 954 String path = AccessController.doPrivileged(new PrivilegedAction<String>() { 955 public String run() { 956 return dir.getAbsolutePath(); 957 } 958 }); 959 960 return path != null && (path.equals("A:\\") || path.equals("B:\\")); 961 } 962 963 /** 964 * Returns a File object constructed from the given path string. 965 */ 966 public File createFileObject(String path) { 967 // Check for missing backslash after drive letter such as "C:" or "C:filename" 968 if (path.length() >= 2 && path.charAt(1) == ':' && Character.isLetter(path.charAt(0))) { 969 if (path.length() == 2) { 970 path += "\\"; 971 } else if (path.charAt(2) != '\\') { 972 path = path.substring(0, 2) + "\\" + path.substring(2); 973 } 974 } 975 return (File) ShellFolder.get("parseDisplayName " + path); 976 } 977 978 private static final String CLSID_NETWORK = "{F02C1A0D-BE21-4350-88B0-7367FC96EF3C}"; 979 private static final String CLSID_COMPUTER = "{20d04fe0-3aea-1069-a2d8-08002b30309d}"; 980 private static final String CLSID_HOMEGROUP = "{67CA7650-96E6-4FDD-BB43-A8E774F73A57}"; 981 private static final String CLSID_FAVORITES = "{323CA680-C24D-4099-B94D-446DD2D7249E}"; 982 private static final String CLSID_LIBRARIES = "{031E4825-7B94-4dc3-B131-E946B44C8DD5}"; 983 private static final String CLSID_QUICK_ACCESS = "{679f85cb-0220-4080-b29b-5540cc05aab6}"; 984 private static final String CLSID_HOMEGROUP_WIN10 = "{B4FB3F98-C1EA-428d-A78A-D1F5659CBA93}"; 985 986 private static final EssentialFolderSection[] ESSENTIAL_FOLDERS = createEssentialFolders(); 987 988 private static EssentialFolderSection[] createEssentialFolders() { 989 List<File> folders = new ArrayList<>(); 990 if (OSInfo.getWindowsVersion().compareTo(OSInfo.WINDOWS_7) <= 0) { 991 folders.add((File) ShellFolder.get(CLSID_FAVORITES)); 992 folders.add((File) ShellFolder.get(CLSID_LIBRARIES)); 993 folders.add((File) ShellFolder.get(CLSID_HOMEGROUP)); 994 folders.add((File) ShellFolder.get(CLSID_COMPUTER)); 995 folders.add((File) ShellFolder.get(CLSID_NETWORK)); 996 } else { 997 folders.add((File) ShellFolder.get(CLSID_QUICK_ACCESS)); 998 folders.add((File) ShellFolder.get(CLSID_COMPUTER)); 999 folders.add((File) ShellFolder.get(CLSID_NETWORK)); 1000 1001 // for some unknown reason, adding both Quick Access and Home Group 1002 // on Windows 10 results in all folders in the FilePane (!) with 1003 // default icon switched their icon to a green person icon. 1004 } 1005 1006 return new EssentialFolderSection[]{ 1007 new EssentialFolderSection(folders.toArray(new File[0])) 1008 }; 1009 } 1010 1011 1012 @Override 1013 public EssentialFolderSection[] getEssentialFolders() { 1014 return Arrays.copyOf(ESSENTIAL_FOLDERS, ESSENTIAL_FOLDERS.length); 1015 } 1016 1017 @SuppressWarnings("serial") // anonymous class 1018 protected File createFileSystemRoot(File f) { 1019 // Problem: Removable drives on Windows return false on f.exists() 1020 // Workaround: Override exists() to always return true. 1021 return new FileSystemRoot(f) { 1022 public boolean exists() { 1023 return true; 1024 } 1025 }; 1026 } 1027 1028 } 1029 1030 /** 1031 * Fallthrough FileSystemView in case we can't determine the OS. 1032 */ 1033 class GenericFileSystemView extends FileSystemView { 1034 1035 private static final String newFolderString = 1036 UIManager.getString("FileChooser.other.newFolder"); 1037 1038 /** 1039 * Creates a new folder with a default folder name. 1040 */ 1041 public File createNewFolder(File containingDir) throws IOException { 1042 if(containingDir == null) { 1043 throw new IOException("Containing directory is null:"); 1044 } 1045 // Using NT's default folder name 1046 File newFolder = createFileObject(containingDir, newFolderString); 1047 1048 if(newFolder.exists()) { 1049 throw new IOException("Directory already exists:" + newFolder.getAbsolutePath()); 1050 } else { 1051 if(!newFolder.mkdirs()) { 1052 throw new IOException(newFolder.getAbsolutePath()); 1053 } 1054 } 1055 return newFolder; 1056 } 1057 1058 @Override 1059 public EssentialFolderSection[] getEssentialFolders() { 1060 return new EssentialFolderSection[]{ 1061 new EssentialFolderSection(getChooserComboBoxFiles()) 1062 }; 1063 } 1064 1065 }