/* * Copyright (c) 2014 Jonas Kalderstam * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ package com.nononsenseapps.filepicker; import android.net.Uri; import android.os.Environment; import android.os.FileObserver; import android.support.v4.content.AsyncTaskLoader; import android.support.v4.content.Loader; import android.support.v7.util.SortedList; import android.support.v7.widget.util.SortedListAdapterCallback; import android.widget.Toast; import java.io.File; public class FilePickerFragment extends AbstractFilePickerFragment { public FilePickerFragment() { } /** * Return true if the path is a directory and not a file. * * @param path */ @Override public boolean isDir(final File path) { return path.isDirectory(); } /** * @param path * @return filename of path */ @Override public String getName(File path) { return path.getName(); } /** * Return the path to the parent directory. Should return the root if * from is root. * * @param from */ @Override public File getParent(final File from) { if (from.getParentFile() != null) { if (from.isFile()) { return getParent(from.getParentFile()); } else { return from.getParentFile(); } } else { return from; } } /** * Convert the path to the type used. * * @param path */ @Override public File getPath(final String path) { return new File(path); } /** * @param path * @return the full path to the file */ @Override public String getFullPath(final File path) { return path.getPath(); } /** * Get the root path (lowest allowed). */ @Override public File getRoot() { return Environment.getExternalStorageDirectory(); } /** * Convert the path to a URI for the return intent * * @param file * @return */ @Override public Uri toUri(final File file) { return Uri.fromFile(file); } /** * Get a loader that lists the Files in the current path, * and monitors changes. */ @Override public Loader> getLoader() { return new AsyncTaskLoader>(getActivity()) { FileObserver fileObserver; @Override public SortedList loadInBackground() { File[] listFiles = currentPath.listFiles(); final int initCap = listFiles == null ? 0 : listFiles.length; SortedList files = new SortedList<>(File.class, new SortedListAdapterCallback(getDummyAdapter()) { @Override public int compare(File lhs, File rhs) { if (lhs.isDirectory() && !rhs.isDirectory()) { return -1; } else if (rhs.isDirectory() && !lhs.isDirectory()) { return 1; } else { return lhs.getName().toLowerCase().compareTo(rhs.getName() .toLowerCase()); } } @Override public boolean areContentsTheSame(File file, File file2) { return file.getAbsolutePath().equals(file2.getAbsolutePath()) && (file.isFile() == file2.isFile()); } @Override public boolean areItemsTheSame(File file, File file2) { return areContentsTheSame(file, file2); } }, initCap); files.beginBatchedUpdates(); if (listFiles != null) { for (java.io.File f : listFiles) { if ((mode == MODE_FILE || mode == MODE_FILE_AND_DIR) || f.isDirectory()) { files.add(f); } } } files.endBatchedUpdates(); return files; } /** * Handles a request to start the Loader. */ @Override protected void onStartLoading() { super.onStartLoading(); // handle if directory does not exist. Fall back to root. if (currentPath == null || !currentPath.isDirectory()) { currentPath = getRoot(); } // Start watching for changes fileObserver = new FileObserver(currentPath.getPath(), FileObserver.CREATE | FileObserver.DELETE | FileObserver.MOVED_FROM | FileObserver.MOVED_TO ) { @Override public void onEvent(int event, String path) { // Reload onContentChanged(); } }; fileObserver.startWatching(); forceLoad(); } /** * Handles a request to completely reset the Loader. */ @Override protected void onReset() { super.onReset(); // Stop watching if (fileObserver != null) { fileObserver.stopWatching(); fileObserver = null; } } }; } /** * Name is validated to be non-null, non-empty and not containing any * slashes. * * @param name The name of the folder the user wishes to create. */ @Override public void onNewFolder(final String name) { File folder = new File(currentPath, name); if (folder.mkdir()) { currentPath = folder; refresh(); } else { Toast.makeText(getActivity(), R.string.create_folder_error, Toast.LENGTH_SHORT).show(); } } }