/* * Copyright 2011 Vaadin Ltd. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ package com.vaadin.data.util; import java.io.File; import java.io.FilenameFilter; import java.io.IOException; import java.io.Serializable; import java.lang.reflect.Method; import java.util.ArrayList; import java.util.Arrays; import java.util.Collection; import java.util.Collections; import java.util.Date; import java.util.Iterator; import java.util.LinkedList; import java.util.List; import com.vaadin.data.Container; import com.vaadin.data.Item; import com.vaadin.data.Property; import com.vaadin.service.FileTypeResolver; import com.vaadin.terminal.Resource; /** * A hierarchical container wrapper for a filesystem. * * @author Vaadin Ltd. * @since 3.0 */ @SuppressWarnings("serial") public class FilesystemContainer implements Container.Hierarchical { /** * String identifier of a file's "name" property. */ public static String PROPERTY_NAME = "Name"; /** * String identifier of a file's "size" property. */ public static String PROPERTY_SIZE = "Size"; /** * String identifier of a file's "icon" property. */ public static String PROPERTY_ICON = "Icon"; /** * String identifier of a file's "last modified" property. */ public static String PROPERTY_LASTMODIFIED = "Last Modified"; /** * List of the string identifiers for the available properties. */ public static Collection FILE_PROPERTIES; private final static Method FILEITEM_LASTMODIFIED; private final static Method FILEITEM_NAME; private final static Method FILEITEM_ICON; private final static Method FILEITEM_SIZE; static { FILE_PROPERTIES = new ArrayList(); FILE_PROPERTIES.add(PROPERTY_NAME); FILE_PROPERTIES.add(PROPERTY_ICON); FILE_PROPERTIES.add(PROPERTY_SIZE); FILE_PROPERTIES.add(PROPERTY_LASTMODIFIED); FILE_PROPERTIES = Collections.unmodifiableCollection(FILE_PROPERTIES); try { FILEITEM_LASTMODIFIED = FileItem.class.getMethod("lastModified", new Class[] {}); FILEITEM_NAME = FileItem.class.getMethod("getName", new Class[] {}); FILEITEM_ICON = FileItem.class.getMethod("getIcon", new Class[] {}); FILEITEM_SIZE = FileItem.class.getMethod("getSize", new Class[] {}); } catch (final NoSuchMethodException e) { throw new RuntimeException( "Internal error finding methods in FilesystemContainer"); } } private File[] roots = new File[] {}; private FilenameFilter filter = null; private boolean recursive = true; /** * Constructs a new FileSystemContainer with the specified file * as the root of the filesystem. The files are included recursively. * * @param root * the root file for the new file-system container. Null values * are ignored. */ public FilesystemContainer(File root) { if (root != null) { roots = new File[] { root }; } } /** * Constructs a new FileSystemContainer with the specified file * as the root of the filesystem. The files are included recursively. * * @param root * the root file for the new file-system container. * @param recursive * should the container recursively contain subdirectories. */ public FilesystemContainer(File root, boolean recursive) { this(root); setRecursive(recursive); } /** * Constructs a new FileSystemContainer with the specified file * as the root of the filesystem. * * @param root * the root file for the new file-system container. * @param extension * the Filename extension (w/o separator) to limit the files in * container. * @param recursive * should the container recursively contain subdirectories. */ public FilesystemContainer(File root, String extension, boolean recursive) { this(root); this.setFilter(extension); setRecursive(recursive); } /** * Constructs a new FileSystemContainer with the specified root * and recursivity status. * * @param root * the root file for the new file-system container. * @param filter * the Filename filter to limit the files in container. * @param recursive * should the container recursively contain subdirectories. */ public FilesystemContainer(File root, FilenameFilter filter, boolean recursive) { this(root); this.setFilter(filter); setRecursive(recursive); } /** * Adds new root file directory. Adds a file to be included as root file * directory in the FilesystemContainer. * * @param root * the File to be added as root directory. Null values are * ignored. */ public void addRoot(File root) { if (root != null) { final File[] newRoots = new File[roots.length + 1]; for (int i = 0; i < roots.length; i++) { newRoots[i] = roots[i]; } newRoots[roots.length] = root; roots = newRoots; } } /** * Tests if the specified Item in the container may have children. Since a * FileSystemContainer contains files and directories, this * method returns true for directory Items only. * * @param itemId * the id of the item. * @return true if the specified Item is a directory, * false otherwise. */ @Override public boolean areChildrenAllowed(Object itemId) { return itemId instanceof File && ((File) itemId).canRead() && ((File) itemId).isDirectory(); } /* * Gets the ID's of all Items who are children of the specified Item. Don't * add a JavaDoc comment here, we use the default documentation from * implemented interface. */ @Override public Collection getChildren(Object itemId) { if (!(itemId instanceof File)) { return Collections.unmodifiableCollection(new LinkedList()); } File[] f; if (filter != null) { f = ((File) itemId).listFiles(filter); } else { f = ((File) itemId).listFiles(); } if (f == null) { return Collections.unmodifiableCollection(new LinkedList()); } final List l = Arrays.asList(f); Collections.sort(l); return Collections.unmodifiableCollection(l); } /* * Gets the parent item of the specified Item. Don't add a JavaDoc comment * here, we use the default documentation from implemented interface. */ @Override public Object getParent(Object itemId) { if (!(itemId instanceof File)) { return null; } return ((File) itemId).getParentFile(); } /* * Tests if the specified Item has any children. Don't add a JavaDoc comment * here, we use the default documentation from implemented interface. */ @Override public boolean hasChildren(Object itemId) { if (!(itemId instanceof File)) { return false; } String[] l; if (filter != null) { l = ((File) itemId).list(filter); } else { l = ((File) itemId).list(); } return (l != null) && (l.length > 0); } /* * Tests if the specified Item is the root of the filesystem. Don't add a * JavaDoc comment here, we use the default documentation from implemented * interface. */ @Override public boolean isRoot(Object itemId) { if (!(itemId instanceof File)) { return false; } for (int i = 0; i < roots.length; i++) { if (roots[i].equals(itemId)) { return true; } } return false; } /* * Gets the ID's of all root Items in the container. Don't add a JavaDoc * comment here, we use the default documentation from implemented * interface. */ @Override public Collection rootItemIds() { File[] f; // in single root case we use children if (roots.length == 1) { if (filter != null) { f = roots[0].listFiles(filter); } else { f = roots[0].listFiles(); } } else { f = roots; } if (f == null) { return Collections.unmodifiableCollection(new LinkedList()); } final List l = Arrays.asList(f); Collections.sort(l); return Collections.unmodifiableCollection(l); } /** * Returns false when conversion from files to directories is * not supported. * * @param itemId * the ID of the item. * @param areChildrenAllowed * the boolean value specifying if the Item can have children or * not. * @return true if the operaton is successful otherwise * false. * @throws UnsupportedOperationException * if the setChildrenAllowed is not supported. */ @Override public boolean setChildrenAllowed(Object itemId, boolean areChildrenAllowed) throws UnsupportedOperationException { throw new UnsupportedOperationException( "Conversion file to/from directory is not supported"); } /** * Returns false when moving files around in the filesystem is * not supported. * * @param itemId * the ID of the item. * @param newParentId * the ID of the Item that's to be the new parent of the Item * identified with itemId. * @return true if the operation is successful otherwise * false. * @throws UnsupportedOperationException * if the setParent is not supported. */ @Override public boolean setParent(Object itemId, Object newParentId) throws UnsupportedOperationException { throw new UnsupportedOperationException("File moving is not supported"); } /* * Tests if the filesystem contains the specified Item. Don't add a JavaDoc * comment here, we use the default documentation from implemented * interface. */ @Override public boolean containsId(Object itemId) { if (!(itemId instanceof File)) { return false; } boolean val = false; // Try to match all roots for (int i = 0; i < roots.length; i++) { try { val |= ((File) itemId).getCanonicalPath().startsWith( roots[i].getCanonicalPath()); } catch (final IOException e) { // Exception ignored } } if (val && filter != null) { val &= filter.accept(((File) itemId).getParentFile(), ((File) itemId).getName()); } return val; } /* * Gets the specified Item from the filesystem. Don't add a JavaDoc comment * here, we use the default documentation from implemented interface. */ @Override public Item getItem(Object itemId) { if (!(itemId instanceof File)) { return null; } return new FileItem((File) itemId); } /** * Internal recursive method to add the files under the specified directory * to the collection. * * @param col * the collection where the found items are added * @param f * the root file where to start adding files */ private void addItemIds(Collection col, File f) { File[] l; if (filter != null) { l = f.listFiles(filter); } else { l = f.listFiles(); } if (l == null) { // File.listFiles returns null if File does not exist or if there // was an IO error (permission denied) return; } final List ll = Arrays.asList(l); Collections.sort(ll); for (final Iterator i = ll.iterator(); i.hasNext();) { final File lf = i.next(); col.add(lf); if (lf.isDirectory()) { addItemIds(col, lf); } } } /* * Gets the IDs of Items in the filesystem. Don't add a JavaDoc comment * here, we use the default documentation from implemented interface. */ @Override public Collection getItemIds() { if (recursive) { final Collection col = new ArrayList(); for (int i = 0; i < roots.length; i++) { addItemIds(col, roots[i]); } return Collections.unmodifiableCollection(col); } else { File[] f; if (roots.length == 1) { if (filter != null) { f = roots[0].listFiles(filter); } else { f = roots[0].listFiles(); } } else { f = roots; } if (f == null) { return Collections .unmodifiableCollection(new LinkedList()); } final List l = Arrays.asList(f); Collections.sort(l); return Collections.unmodifiableCollection(l); } } /** * Gets the specified property of the specified file Item. The available * file properties are "Name", "Size" and "Last Modified". If propertyId is * not one of those, null is returned. * * @param itemId * the ID of the file whose property is requested. * @param propertyId * the property's ID. * @return the requested property's value, or null */ @Override public Property getContainerProperty(Object itemId, Object propertyId) { if (!(itemId instanceof File)) { return null; } if (propertyId.equals(PROPERTY_NAME)) { return new MethodProperty(getType(propertyId), new FileItem((File) itemId), FILEITEM_NAME, null); } if (propertyId.equals(PROPERTY_ICON)) { return new MethodProperty(getType(propertyId), new FileItem((File) itemId), FILEITEM_ICON, null); } if (propertyId.equals(PROPERTY_SIZE)) { return new MethodProperty(getType(propertyId), new FileItem((File) itemId), FILEITEM_SIZE, null); } if (propertyId.equals(PROPERTY_LASTMODIFIED)) { return new MethodProperty(getType(propertyId), new FileItem((File) itemId), FILEITEM_LASTMODIFIED, null); } return null; } /** * Gets the collection of available file properties. * * @return Unmodifiable collection containing all available file properties. */ @Override public Collection getContainerPropertyIds() { return FILE_PROPERTIES; } /** * Gets the specified property's data type. "Name" is a String, * "Size" is a Long, "Last Modified" is a Date. If * propertyId is not one of those, null is returned. * * @param propertyId * the ID of the property whose type is requested. * @return data type of the requested property, or null */ @Override public Class getType(Object propertyId) { if (propertyId.equals(PROPERTY_NAME)) { return String.class; } if (propertyId.equals(PROPERTY_ICON)) { return Resource.class; } if (propertyId.equals(PROPERTY_SIZE)) { return Long.class; } if (propertyId.equals(PROPERTY_LASTMODIFIED)) { return Date.class; } return null; } /** * Internal method to recursively calculate the number of files under a root * directory. * * @param f * the root to start counting from. */ private int getFileCounts(File f) { File[] l; if (filter != null) { l = f.listFiles(filter); } else { l = f.listFiles(); } if (l == null) { return 0; } int ret = l.length; for (int i = 0; i < l.length; i++) { if (l[i].isDirectory()) { ret += getFileCounts(l[i]); } } return ret; } /** * Gets the number of Items in the container. In effect, this is the * combined amount of files and directories. * * @return Number of Items in the container. */ @Override public int size() { if (recursive) { int counts = 0; for (int i = 0; i < roots.length; i++) { counts += getFileCounts(roots[i]); } return counts; } else { File[] f; if (roots.length == 1) { if (filter != null) { f = roots[0].listFiles(filter); } else { f = roots[0].listFiles(); } } else { f = roots; } if (f == null) { return 0; } return f.length; } } /** * A Item wrapper for files in a filesystem. * * @author Vaadin Ltd. * @since 3.0 */ public class FileItem implements Item { /** * The wrapped file. */ private final File file; /** * Constructs a FileItem from a existing file. */ private FileItem(File file) { this.file = file; } /* * Gets the specified property of this file. Don't add a JavaDoc comment * here, we use the default documentation from implemented interface. */ @Override public Property getItemProperty(Object id) { return getContainerProperty(file, id); } /* * Gets the IDs of all properties available for this item Don't add a * JavaDoc comment here, we use the default documentation from * implemented interface. */ @Override public Collection getItemPropertyIds() { return getContainerPropertyIds(); } /** * Calculates a integer hash-code for the Property that's unique inside * the Item containing the Property. Two different Properties inside the * same Item contained in the same list always have different * hash-codes, though Properties in different Items may have identical * hash-codes. * * @return A locally unique hash-code as integer */ @Override public int hashCode() { return file.hashCode() ^ FilesystemContainer.this.hashCode(); } /** * Tests if the given object is the same as the this object. Two * Properties got from an Item with the same ID are equal. * * @param obj * an object to compare with this object. * @return true if the given object is the same as this * object, false if not */ @Override public boolean equals(Object obj) { if (obj == null || !(obj instanceof FileItem)) { return false; } final FileItem fi = (FileItem) obj; return fi.getHost() == getHost() && fi.file.equals(file); } /** * Gets the host of this file. */ private FilesystemContainer getHost() { return FilesystemContainer.this; } /** * Gets the last modified date of this file. * * @return Date */ public Date lastModified() { return new Date(file.lastModified()); } /** * Gets the name of this file. * * @return file name of this file. */ public String getName() { return file.getName(); } /** * Gets the icon of this file. * * @return the icon of this file. */ public Resource getIcon() { return FileTypeResolver.getIcon(file); } /** * Gets the size of this file. * * @return size */ public long getSize() { if (file.isDirectory()) { return 0; } return file.length(); } /** * @see java.lang.Object#toString() */ @Override public String toString() { if ("".equals(file.getName())) { return file.getAbsolutePath(); } return file.getName(); } /** * Filesystem container does not support adding new properties. * * @see com.vaadin.data.Item#addItemProperty(Object, Property) */ @Override public boolean addItemProperty(Object id, Property property) throws UnsupportedOperationException { throw new UnsupportedOperationException("Filesystem container " + "does not support adding new properties"); } /** * Filesystem container does not support removing properties. * * @see com.vaadin.data.Item#removeItemProperty(Object) */ @Override public boolean removeItemProperty(Object id) throws UnsupportedOperationException { throw new UnsupportedOperationException( "Filesystem container does not support property removal"); } } /** * Generic file extension filter for displaying only files having certain * extension. * * @author Vaadin Ltd. * @since 3.0 */ public class FileExtensionFilter implements FilenameFilter, Serializable { private final String filter; /** * Constructs a new FileExtensionFilter using given extension. * * @param fileExtension * the File extension without the separator (dot). */ public FileExtensionFilter(String fileExtension) { filter = "." + fileExtension; } /** * Allows only files with the extension and directories. * * @see java.io.FilenameFilter#accept(File, String) */ @Override public boolean accept(File dir, String name) { if (name.endsWith(filter)) { return true; } return new File(dir, name).isDirectory(); } } /** * Returns the file filter used to limit the files in this container. * * @return Used filter instance or null if no filter is assigned. */ public FilenameFilter getFilter() { return filter; } /** * Sets the file filter used to limit the files in this container. * * @param filter * The filter to set. null disables filtering. */ public void setFilter(FilenameFilter filter) { this.filter = filter; } /** * Sets the file filter used to limit the files in this container. * * @param extension * the Filename extension (w/o separator) to limit the files in * container. */ public void setFilter(String extension) { filter = new FileExtensionFilter(extension); } /** * Is this container recursive filesystem. * * @return true if container is recursive, false * otherwise. */ public boolean isRecursive() { return recursive; } /** * Sets the container recursive property. Set this to false to limit the * files directly under the root file. *

* Note : This is meaningful only if the root really is a directory. *

* * @param recursive * the New value for recursive property. */ public void setRecursive(boolean recursive) { this.recursive = recursive; } /* * (non-Javadoc) * * @see com.vaadin.data.Container#addContainerProperty(java.lang.Object, * java.lang.Class, java.lang.Object) */ @Override public boolean addContainerProperty(Object propertyId, Class type, Object defaultValue) throws UnsupportedOperationException { throw new UnsupportedOperationException( "File system container does not support this operation"); } /* * (non-Javadoc) * * @see com.vaadin.data.Container#addItem() */ @Override public Object addItem() throws UnsupportedOperationException { throw new UnsupportedOperationException( "File system container does not support this operation"); } /* * (non-Javadoc) * * @see com.vaadin.data.Container#addItem(java.lang.Object) */ @Override public Item addItem(Object itemId) throws UnsupportedOperationException { throw new UnsupportedOperationException( "File system container does not support this operation"); } /* * (non-Javadoc) * * @see com.vaadin.data.Container#removeAllItems() */ @Override public boolean removeAllItems() throws UnsupportedOperationException { throw new UnsupportedOperationException( "File system container does not support this operation"); } /* * (non-Javadoc) * * @see com.vaadin.data.Container#removeItem(java.lang.Object) */ @Override public boolean removeItem(Object itemId) throws UnsupportedOperationException { throw new UnsupportedOperationException( "File system container does not support this operation"); } /* * (non-Javadoc) * * @see com.vaadin.data.Container#removeContainerProperty(java.lang.Object ) */ @Override public boolean removeContainerProperty(Object propertyId) throws UnsupportedOperationException { throw new UnsupportedOperationException( "File system container does not support this operation"); } }