net.sf.javagimmicks.io.FileTraverser Maven / Gradle / Ivy
package net.sf.javagimmicks.io;
import java.io.File;
import java.io.FileFilter;
/**
* A simple visitor-style implementation for traversing {@link File}s within the
* file system.
*/
public class FileTraverser
{
private final File _root;
private TypeFilter _typeFilter = TypeFilter.FILE;
private FileFilter _fileFilter;
private boolean _recursive = true;
/**
* Creates a new instance for the given root folder.
*
* @param root
* the root folder where to start traversing (must be a non-null
* {@link File} pointing to a directory)
* @throws IllegalArgumentException
* if the given root is null
or not a directory
*/
public FileTraverser(final File root)
{
if (root == null)
{
throw new IllegalArgumentException("Root folder is null!");
}
if (!root.isDirectory())
{
throw new IllegalArgumentException("Given file is not a directory!");
}
_root = root;
}
/**
* Starts a new run with the given settins and using the given
* {@link FileVisitor} to report visited {@link File}s.
*
* @param visitor
* the {@link FileVisitor} where to report visited {@link File}s
*/
public void run(final FileVisitor visitor)
{
if (visitor == null)
{
return;
}
scan(_root, visitor);
}
/**
* Returns the current {@link TypeFilter} which determines what types of
* {@link File}s should be visited (files and/or folders).
*
* @return the current {@link TypeFilter}
*/
public TypeFilter getTypeFilter()
{
return _typeFilter;
}
/**
* Sets a new {@link TypeFilter}.
*
* @param typeFilter
* the new {@link TypeFilter} to set
* @throws IllegalArgumentException
* if the given {@link TypeFilter} is null
* @see FileTraverser#getTypeFilter()
*/
public void setTypeFilter(final TypeFilter typeFilter)
{
if (typeFilter == null)
{
throw new IllegalArgumentException("TypeFilter may not be null!");
}
this._typeFilter = typeFilter;
}
/**
* Returns the current {@link FileFilter} which determines what files and/or
* folders should be visited.
*
* Note: folders not included by this {@link FileFilter} will be
* skipped completely by {@link #run(FileVisitor)} calls (not visited and not
* recursed into if {@link #isRecursive()} is set).
*
* @return the current {@link FileFilter}
*/
public FileFilter getFileFilter()
{
return _fileFilter;
}
/**
* Sets a new {@link FileFilter}.
*
* @param fileFilter
* the new {@link FileFilter} - may be null
* @see #getFileFilter()
*/
public void setFileFilter(final FileFilter fileFilter)
{
this._fileFilter = fileFilter;
}
/**
* Returns if calls to {@link #run(FileVisitor)} will recurse into
* sub-folders.
*
* @return if calls to {@link #run(FileVisitor)} will recurse into
* sub-folders
*/
public boolean isRecursive()
{
return _recursive;
}
/**
* Sets if if calls to {@link #run(FileVisitor)} will recurse into
* sub-folders.
*
* @param recursive
* if calls to {@link #run(FileVisitor)} will recurse into
* sub-folders
*/
public void setRecursive(final boolean recursive)
{
this._recursive = recursive;
}
private void scan(final File directory, final FileVisitor visitor)
{
final File[] children = directory.listFiles(_fileFilter);
for (final File child : children)
{
if (child.isDirectory())
{
if (_typeFilter._acceptsFolders)
{
visitor.visit(child);
}
if (_recursive)
{
scan(child, visitor);
}
}
}
if (_typeFilter._acceptsFiles)
{
for (final File child : children)
{
if (child.isFile())
{
visitor.visit(child);
}
}
}
}
/**
* A simple visitor for {@link File} instances.
*/
public static interface FileVisitor
{
/**
* Callback method for a visited {@link File}
*
* @param file
* the {@link File} that was visited
*/
void visit(File file);
}
/**
* Determines what types of {@link File}s to report to
* {@link FileVisitor#visit(File)} within calls to
* {@link FileTraverser#run(FileVisitor)}.
*/
public static enum TypeFilter
{
/**
* Visit files only.
*/
FILE(true, false),
/**
* Visit folders only.
*/
FOLDER(false, true),
/**
* Visit files and folders.
*/
BOTH(true, true);
private final boolean _acceptsFiles;
private final boolean _acceptsFolders;
private TypeFilter(final boolean acceptsFiles, final boolean acceptsFolders)
{
_acceptsFiles = acceptsFiles;
_acceptsFolders = acceptsFolders;
}
};
}