mirror of
				https://github.com/fankes/termux-app.git
				synced 2025-10-26 13:49:20 +08:00 
			
		
		
		
	Expose files through the Storage Access Framework
This allows e.g. external editors to edit files in the Termux home folder. Fixes #79.
This commit is contained in:
		| @@ -0,0 +1,235 @@ | ||||
| package com.termux.filepicker; | ||||
|  | ||||
| import android.content.res.AssetFileDescriptor; | ||||
| import android.database.Cursor; | ||||
| import android.database.MatrixCursor; | ||||
| import android.graphics.Point; | ||||
| import android.os.CancellationSignal; | ||||
| import android.os.ParcelFileDescriptor; | ||||
| import android.provider.DocumentsContract.Document; | ||||
| import android.provider.DocumentsContract.Root; | ||||
| import android.provider.DocumentsProvider; | ||||
| import android.webkit.MimeTypeMap; | ||||
|  | ||||
| import com.termux.R; | ||||
| import com.termux.app.TermuxService; | ||||
|  | ||||
| import java.io.File; | ||||
| import java.io.FileNotFoundException; | ||||
| import java.io.IOException; | ||||
| import java.util.Collections; | ||||
| import java.util.LinkedList; | ||||
|  | ||||
| /** | ||||
|  * A document provider for the Storage Access Framework which exposes the files in the | ||||
|  * $HOME/ folder to other apps. | ||||
|  */ | ||||
| public class TermuxDocumentsProvider extends DocumentsProvider { | ||||
|  | ||||
|     private static final String ALL_MIME_TYPES = "*/*"; | ||||
|  | ||||
|     private static final File BASE_DIR = new File(TermuxService.HOME_PATH); | ||||
|  | ||||
|  | ||||
|     // The default columns to return information about a root if no specific | ||||
|     // columns are requested in a query. | ||||
|     private static final String[] DEFAULT_ROOT_PROJECTION = new String[]{ | ||||
|         Root.COLUMN_ROOT_ID, | ||||
|         Root.COLUMN_MIME_TYPES, | ||||
|         Root.COLUMN_FLAGS, | ||||
|         Root.COLUMN_ICON, | ||||
|         Root.COLUMN_TITLE, | ||||
|         Root.COLUMN_SUMMARY, | ||||
|         Root.COLUMN_DOCUMENT_ID, | ||||
|         Root.COLUMN_AVAILABLE_BYTES | ||||
|     }; | ||||
|  | ||||
|     // The default columns to return information about a document if no specific | ||||
|     // columns are requested in a query. | ||||
|     private static final String[] DEFAULT_DOCUMENT_PROJECTION = new String[]{ | ||||
|         Document.COLUMN_DOCUMENT_ID, | ||||
|         Document.COLUMN_MIME_TYPE, | ||||
|         Document.COLUMN_DISPLAY_NAME, | ||||
|         Document.COLUMN_LAST_MODIFIED, | ||||
|         Document.COLUMN_FLAGS, | ||||
|         Document.COLUMN_SIZE | ||||
|     }; | ||||
|  | ||||
|     @Override | ||||
|     public Cursor queryRoots(String[] projection) throws FileNotFoundException { | ||||
|         final MatrixCursor result = new MatrixCursor(projection != null ? projection : DEFAULT_ROOT_PROJECTION); | ||||
|         @SuppressWarnings("ConstantConditions") final String applicationName = getContext().getString(R.string.application_name); | ||||
|  | ||||
|         final MatrixCursor.RowBuilder row = result.newRow(); | ||||
|         row.add(Root.COLUMN_ROOT_ID, getDocIdForFile(BASE_DIR)); | ||||
|         row.add(Root.COLUMN_DOCUMENT_ID, getDocIdForFile(BASE_DIR)); | ||||
|         row.add(Root.COLUMN_SUMMARY, null); | ||||
|         row.add(Root.COLUMN_FLAGS, Root.FLAG_SUPPORTS_CREATE | Root.FLAG_SUPPORTS_RECENTS | Root.FLAG_SUPPORTS_SEARCH); | ||||
|         row.add(Root.COLUMN_TITLE, applicationName); | ||||
|         row.add(Root.COLUMN_MIME_TYPES, ALL_MIME_TYPES); | ||||
|         row.add(Root.COLUMN_AVAILABLE_BYTES, BASE_DIR.getFreeSpace()); | ||||
|         row.add(Root.COLUMN_ICON, R.drawable.ic_launcher); | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public Cursor queryDocument(String documentId, String[] projection) throws FileNotFoundException { | ||||
|         final MatrixCursor result = new MatrixCursor(projection != null ? projection : DEFAULT_DOCUMENT_PROJECTION); | ||||
|         includeFile(result, documentId, null); | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public Cursor queryChildDocuments(String parentDocumentId, String[] projection, String sortOrder) throws FileNotFoundException { | ||||
|         final MatrixCursor result = new MatrixCursor(projection != null ? projection : DEFAULT_DOCUMENT_PROJECTION); | ||||
|         final File parent = getFileForDocId(parentDocumentId); | ||||
|         for (File file : parent.listFiles()) { | ||||
|             if (!file.getName().startsWith(".")) { | ||||
|                 includeFile(result, null, file); | ||||
|             } | ||||
|         } | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public ParcelFileDescriptor openDocument(final String documentId, String mode, CancellationSignal signal) throws FileNotFoundException { | ||||
|         final File file = getFileForDocId(documentId); | ||||
|         final int accessMode = ParcelFileDescriptor.parseMode(mode); | ||||
|         return ParcelFileDescriptor.open(file, accessMode); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public AssetFileDescriptor openDocumentThumbnail(String documentId, Point sizeHint, CancellationSignal signal) throws FileNotFoundException { | ||||
|         final File file = getFileForDocId(documentId); | ||||
|         final ParcelFileDescriptor pfd = ParcelFileDescriptor.open(file, ParcelFileDescriptor.MODE_READ_ONLY); | ||||
|         return new AssetFileDescriptor(pfd, 0, file.length()); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean onCreate() { | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void deleteDocument(String documentId) throws FileNotFoundException { | ||||
|         File file = getFileForDocId(documentId); | ||||
|         if (!file.delete()) { | ||||
|             throw new FileNotFoundException("Failed to delete document with id " + documentId); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public String getDocumentType(String documentId) throws FileNotFoundException { | ||||
|         File file = getFileForDocId(documentId); | ||||
|         return getMimeType(file); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public Cursor querySearchDocuments(String rootId, String query, String[] projection) throws FileNotFoundException { | ||||
|         final MatrixCursor result = new MatrixCursor(projection != null ? projection : DEFAULT_DOCUMENT_PROJECTION); | ||||
|         final File parent = getFileForDocId(rootId); | ||||
|  | ||||
|         // This example implementation searches file names for the query and doesn't rank search | ||||
|         // results, so we can stop as soon as we find a sufficient number of matches.  Other | ||||
|         // implementations might rank results and use other data about files, rather than the file | ||||
|         // name, to produce a match. | ||||
|         final LinkedList<File> pending = new LinkedList<>(); | ||||
|         pending.add(parent); | ||||
|  | ||||
|         final int MAX_SEARCH_RESULTS = 50; | ||||
|         while (!pending.isEmpty() && result.getCount() < MAX_SEARCH_RESULTS) { | ||||
|             final File file = pending.removeFirst(); | ||||
|             // Avoid folders outside the $HOME folders linked in to symlinks (to avoid e.g. search | ||||
|             // through the whole SD card). | ||||
|             boolean isInsideHome; | ||||
|             try { | ||||
|                 isInsideHome = file.getCanonicalPath().startsWith(TermuxService.HOME_PATH); | ||||
|             } catch (IOException e) { | ||||
|                 isInsideHome = true; | ||||
|             } | ||||
|             final boolean isHidden = file.getName().startsWith("."); | ||||
|             if (isInsideHome && !isHidden) { | ||||
|                 if (file.isDirectory()) { | ||||
|                     Collections.addAll(pending, file.listFiles()); | ||||
|                 } else { | ||||
|                     if (file.getName().toLowerCase().contains(query)) { | ||||
|                         includeFile(result, null, file); | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get the document id given a file. This document id must be consistent across time as other | ||||
|      * applications may save the ID and use it to reference documents later. | ||||
|      * <p> | ||||
|      * The reverse of @{link #getFileForDocId}. | ||||
|      */ | ||||
|     private static String getDocIdForFile(File file) { | ||||
|         return file.getAbsolutePath(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get the file given a document id (the reverse of {@link #getFileForDocId(String)}). | ||||
|      */ | ||||
|     private static File getFileForDocId(String docId) throws FileNotFoundException { | ||||
|         final File f = new File(docId); | ||||
|         if (!f.exists()) throw new FileNotFoundException(f.getAbsolutePath() + " not found"); | ||||
|         return f; | ||||
|     } | ||||
|  | ||||
|     private static String getMimeType(File file) { | ||||
|         if (file.isDirectory()) { | ||||
|             return Document.MIME_TYPE_DIR; | ||||
|         } else { | ||||
|             final String name = file.getName(); | ||||
|             final int lastDot = name.lastIndexOf('.'); | ||||
|             if (lastDot >= 0) { | ||||
|                 final String extension = name.substring(lastDot + 1); | ||||
|                 final String mime = MimeTypeMap.getSingleton().getMimeTypeFromExtension(extension); | ||||
|                 if (mime != null) return mime; | ||||
|             } | ||||
|             return "application/octet-stream"; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Add a representation of a file to a cursor. | ||||
|      * | ||||
|      * @param result the cursor to modify | ||||
|      * @param docId  the document ID representing the desired file (may be null if given file) | ||||
|      * @param file   the File object representing the desired file (may be null if given docID) | ||||
|      */ | ||||
|     private void includeFile(MatrixCursor result, String docId, File file) | ||||
|         throws FileNotFoundException { | ||||
|         if (docId == null) { | ||||
|             docId = getDocIdForFile(file); | ||||
|         } else { | ||||
|             file = getFileForDocId(docId); | ||||
|         } | ||||
|  | ||||
|         int flags = 0; | ||||
|         if (file.isDirectory()) { | ||||
|             if (file.isDirectory() && file.canWrite()) flags |= Document.FLAG_DIR_SUPPORTS_CREATE; | ||||
|         } else if (file.canWrite()) { | ||||
|             flags |= Document.FLAG_SUPPORTS_WRITE | Document.FLAG_SUPPORTS_DELETE; | ||||
|         } | ||||
|  | ||||
|         final String displayName = file.getName(); | ||||
|         final String mimeType = getMimeType(file); | ||||
|         if (mimeType.startsWith("image/")) flags |= Document.FLAG_SUPPORTS_THUMBNAIL; | ||||
|  | ||||
|         final MatrixCursor.RowBuilder row = result.newRow(); | ||||
|         row.add(Document.COLUMN_DOCUMENT_ID, docId); | ||||
|         row.add(Document.COLUMN_DISPLAY_NAME, displayName); | ||||
|         row.add(Document.COLUMN_SIZE, file.length()); | ||||
|         row.add(Document.COLUMN_MIME_TYPE, mimeType); | ||||
|         row.add(Document.COLUMN_LAST_MODIFIED, file.lastModified()); | ||||
|         row.add(Document.COLUMN_FLAGS, flags); | ||||
|         row.add(Document.COLUMN_ICON, R.drawable.ic_launcher); | ||||
|     } | ||||
|  | ||||
| } | ||||
		Reference in New Issue
	
	Block a user