mirror of
				https://github.com/fankes/termux-app.git
				synced 2025-10-26 21:59:21 +08:00 
			
		
		
		
	Make it possible to receive files
The files are saved to $HOME/downloads/, after which the user may choose to open the downloads/ folder or edit the file with the $HOME/bin/termux-file-editor program. It's also possible to receive URL:s, in which case the $HOME/bin/termux-url-opener program will be called.
This commit is contained in:
		| @@ -13,14 +13,17 @@ import android.widget.EditText; | ||||
| import android.widget.LinearLayout; | ||||
| import android.widget.TextView; | ||||
|  | ||||
| final class DialogUtils { | ||||
| public final class DialogUtils { | ||||
|  | ||||
| 	public interface TextSetListener { | ||||
| 		void onTextSet(String text); | ||||
| 	} | ||||
|  | ||||
| 	static void textInput(Activity activity, int titleText, int positiveButtonText, String initialText, final TextSetListener onPositive, | ||||
| 										 int neutralButtonText, final TextSetListener onNeutral) { | ||||
| 	public static void textInput(Activity activity, int titleText, String initialText, | ||||
|                                  int positiveButtonText, final TextSetListener onPositive, | ||||
|                                  int neutralButtonText, final TextSetListener onNeutral, | ||||
|                                  int negativeButtonText, final TextSetListener onNegative, | ||||
|                                  final DialogInterface.OnDismissListener onDismiss) { | ||||
| 		final EditText input = new EditText(activity); | ||||
| 		input.setSingleLine(); | ||||
| 		if (initialText != null) { | ||||
| @@ -57,23 +60,32 @@ final class DialogUtils { | ||||
| 					public void onClick(DialogInterface d, int whichButton) { | ||||
| 						onPositive.onTextSet(input.getText().toString()); | ||||
| 				} | ||||
| 			}) | ||||
| 				.setNegativeButton(android.R.string.cancel, null); | ||||
|  | ||||
| 		if (onNeutral != null) { | ||||
| 			builder.setNeutralButton(neutralButtonText, new DialogInterface.OnClickListener() { | ||||
| 				@Override | ||||
| 				public void onClick(DialogInterface dialog, int which) { | ||||
| 					onNeutral.onTextSet(input.getText().toString()); | ||||
| 				} | ||||
| 			}); | ||||
| 		} | ||||
|  | ||||
|         if (onNeutral != null) { | ||||
|             builder.setNeutralButton(neutralButtonText, new DialogInterface.OnClickListener() { | ||||
|                 @Override | ||||
|                 public void onClick(DialogInterface dialog, int which) { | ||||
|                     onNeutral.onTextSet(input.getText().toString()); | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
|  | ||||
|         if (onNegative == null) { | ||||
|             builder.setNegativeButton(android.R.string.cancel, null); | ||||
|         } else { | ||||
|             builder.setNegativeButton(negativeButtonText, new DialogInterface.OnClickListener() { | ||||
|                 @Override | ||||
|                 public void onClick(DialogInterface dialog, int which) { | ||||
|                     onNegative.onTextSet(input.getText().toString()); | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
|  | ||||
|         if (onDismiss != null) builder.setOnDismissListener(onDismiss); | ||||
|  | ||||
| 		dialogHolder[0] = builder.create(); | ||||
| 		if ((activity.getResources().getConfiguration().hardKeyboardHidden & Configuration.HARDKEYBOARDHIDDEN_YES) == 0) { | ||||
| 			// Show soft keyboard unless hardware keyboard available. | ||||
| 			dialogHolder[0].getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_VISIBLE); | ||||
| 		} | ||||
|         dialogHolder[0].setCanceledOnTouchOutside(false); | ||||
| 		dialogHolder[0].show(); | ||||
| 	} | ||||
|  | ||||
|   | ||||
| @@ -287,28 +287,28 @@ public final class TermuxActivity extends Activity implements ServiceConnection | ||||
| 			} | ||||
| 		}); | ||||
|  | ||||
| 		newSessionButton.setOnLongClickListener(new OnLongClickListener() { | ||||
| 			@Override | ||||
| 			public boolean onLongClick(View v) { | ||||
| 				Resources res = getResources(); | ||||
| 				DialogUtils.textInput(TermuxActivity.this, R.string.session_new_named_title, R.string.session_new_named_positive_button, null, | ||||
| 					new DialogUtils.TextSetListener() { | ||||
| 						@Override | ||||
| 						public void onTextSet(String text) { | ||||
| 							addNewSession(false, text); | ||||
| 						} | ||||
| 					}, R.string.new_session_failsafe, new DialogUtils.TextSetListener() { | ||||
| 						@Override | ||||
| 						public void onTextSet(String text) { | ||||
| 							addNewSession(true, text); | ||||
| 						} | ||||
| 					} | ||||
| 				); | ||||
| 				return true; | ||||
| 			} | ||||
| 		}); | ||||
|         newSessionButton.setOnLongClickListener(new OnLongClickListener() { | ||||
|             @Override | ||||
|             public boolean onLongClick(View v) { | ||||
|                 Resources res = getResources(); | ||||
|                 DialogUtils.textInput(TermuxActivity.this, R.string.session_new_named_title, null, R.string.session_new_named_positive_button, | ||||
|                         new DialogUtils.TextSetListener() { | ||||
|                             @Override | ||||
|                             public void onTextSet(String text) { | ||||
|                                 addNewSession(false, text); | ||||
|                             } | ||||
|                         }, R.string.new_session_failsafe, new DialogUtils.TextSetListener() { | ||||
|                             @Override | ||||
|                             public void onTextSet(String text) { | ||||
|                                 addNewSession(true, text); | ||||
|                             } | ||||
|                         } | ||||
|                         , -1, null, null); | ||||
|                 return true; | ||||
|             } | ||||
|         }); | ||||
|  | ||||
| 		findViewById(R.id.toggle_keyboard_button).setOnClickListener(new OnClickListener() { | ||||
|         findViewById(R.id.toggle_keyboard_button).setOnClickListener(new OnClickListener() { | ||||
| 			@Override | ||||
| 			public void onClick(View v) { | ||||
| 				InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE); | ||||
| @@ -492,14 +492,13 @@ public final class TermuxActivity extends Activity implements ServiceConnection | ||||
|  | ||||
| 	@SuppressLint("InflateParams") | ||||
| 	void renameSession(final TerminalSession sessionToRename) { | ||||
| 		DialogUtils.textInput(this, R.string.session_rename_title, R.string.session_rename_positive_button, sessionToRename.mSessionName, | ||||
| 				new DialogUtils.TextSetListener() { | ||||
| 					@Override | ||||
| 					public void onTextSet(String text) { | ||||
| 						sessionToRename.mSessionName = text; | ||||
| 					} | ||||
| 				}, -1, null); | ||||
| 	} | ||||
|         DialogUtils.textInput(this, R.string.session_rename_title, sessionToRename.mSessionName, R.string.session_rename_positive_button, new DialogUtils.TextSetListener() { | ||||
|             @Override | ||||
|             public void onTextSet(String text) { | ||||
|                 sessionToRename.mSessionName = text; | ||||
|             } | ||||
|         }, -1, null, -1, null, null); | ||||
|     } | ||||
|  | ||||
| 	@Override | ||||
| 	public void onServiceDisconnected(ComponentName name) { | ||||
|   | ||||
| @@ -0,0 +1,226 @@ | ||||
| package com.termux.filepicker; | ||||
|  | ||||
| import android.app.Activity; | ||||
| import android.app.AlertDialog; | ||||
| import android.content.DialogInterface; | ||||
| import android.content.Intent; | ||||
| import android.database.Cursor; | ||||
| import android.net.Uri; | ||||
| import android.provider.OpenableColumns; | ||||
| import android.util.Log; | ||||
| import android.util.MutableBoolean; | ||||
| import android.util.Patterns; | ||||
|  | ||||
| import com.termux.R; | ||||
| import com.termux.app.DialogUtils; | ||||
| import com.termux.app.TermuxService; | ||||
|  | ||||
| import java.io.ByteArrayInputStream; | ||||
| import java.io.File; | ||||
| import java.io.FileInputStream; | ||||
| import java.io.FileNotFoundException; | ||||
| import java.io.FileOutputStream; | ||||
| import java.io.IOException; | ||||
| import java.io.InputStream; | ||||
| import java.nio.charset.StandardCharsets; | ||||
|  | ||||
| public class TermuxFileReceiverActivity extends Activity { | ||||
|  | ||||
|     static final String TERMUX_RECEIVEDIR = TermuxService.FILES_PATH + "/home/downloads"; | ||||
|     static final String EDITOR_PROGRAM = TermuxService.HOME_PATH + "/bin/termux-file-editor"; | ||||
|     static final String URL_OPENER_PROGRAM = TermuxService.HOME_PATH + "/bin/termux-url-opener"; | ||||
|  | ||||
|     /** | ||||
|      * If the activity should be finished when the name input dialog is dismissed. This is disabled | ||||
|      * before showing an error dialog, since the act of showing the error dialog will cause the | ||||
|      * name input dialog to be implicitly dismissed, and we do not want to finish the activity directly | ||||
|      * when showing the error dialog. | ||||
|      */ | ||||
|     private boolean mFinishOnDismissNameDialog = true; | ||||
|  | ||||
|     @Override | ||||
|     protected void onResume() { | ||||
|         super.onResume(); | ||||
|  | ||||
|         final Intent intent = getIntent(); | ||||
|         final String action = intent.getAction(); | ||||
|         final String type = intent.getType(); | ||||
|         final String scheme = intent.getScheme(); | ||||
|  | ||||
|         if (intent.getExtras() == null) { | ||||
|             Log.e("termux", "NULL EXTRAS"); | ||||
|         } else { | ||||
|             for (String key : intent.getExtras().keySet()) { | ||||
|                 Object value = intent.getExtras().get(key); | ||||
|                 Log.d("termux", String.format("Extra %s %s (%s)", key, | ||||
|                     value.toString(), value.getClass().getName())); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (Intent.ACTION_SEND.equals(action) && type != null) { | ||||
|             final String sharedText = intent.getStringExtra(Intent.EXTRA_TEXT); | ||||
|             final Uri sharedUri = intent.getParcelableExtra(Intent.EXTRA_STREAM); | ||||
|  | ||||
|             if (sharedText != null) { | ||||
|                 if (Patterns.WEB_URL.matcher(sharedText).matches()) { | ||||
|                     handleUrlAndFinish(sharedText); | ||||
|                 } else { | ||||
|                     String subject = intent.getStringExtra(Intent.EXTRA_SUBJECT); | ||||
|                     if (subject == null) subject = intent.getStringExtra(Intent.EXTRA_TITLE); | ||||
|                     if (subject != null) subject += ".txt"; | ||||
|                     promptNameAndSave(new ByteArrayInputStream(sharedText.getBytes(StandardCharsets.UTF_8)), subject); | ||||
|                 } | ||||
|             } else if (sharedUri != null) { | ||||
|                 handleContentUri(sharedUri, intent.getStringExtra(Intent.EXTRA_TITLE)); | ||||
|             } else { | ||||
|                 showErrorDialogAndQuit("Send action without content - nothing to save."); | ||||
|             } | ||||
|         } else if (scheme.equals("content")) { | ||||
|             handleContentUri(intent.getData(), intent.getStringExtra(Intent.EXTRA_TITLE)); | ||||
|         } else if (scheme.equals("file")) { | ||||
|             // When e.g. clicking on a downloaded apk: | ||||
|             String path = intent.getData().getPath(); | ||||
|             File file = new File(path); | ||||
|             try { | ||||
|                 FileInputStream in = new FileInputStream(file); | ||||
|                 promptNameAndSave(in, file.getName()); | ||||
|             } catch (FileNotFoundException e) { | ||||
|                 showErrorDialogAndQuit("Cannot open file: " + e.getMessage() + "."); | ||||
|             } | ||||
|         } else { | ||||
|             showErrorDialogAndQuit("Unhandled scheme: " + intent.getScheme() + "."); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void showErrorDialogAndQuit(String message) { | ||||
|         mFinishOnDismissNameDialog = false; | ||||
|         new AlertDialog.Builder(this).setMessage(message).setOnDismissListener(new DialogInterface.OnDismissListener() { | ||||
|             @Override | ||||
|             public void onDismiss(DialogInterface dialog) { | ||||
|                 finish(); | ||||
|             } | ||||
|         }).setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() { | ||||
|             @Override | ||||
|             public void onClick(DialogInterface dialog, int which) { | ||||
|                 finish(); | ||||
|             } | ||||
|         }).show(); | ||||
|     } | ||||
|  | ||||
|     void handleContentUri(final Uri uri, String subjectFromIntent) { | ||||
|         try { | ||||
|             String attachmentFileName = null; | ||||
|  | ||||
|             String[] projection = new String[]{OpenableColumns.DISPLAY_NAME}; | ||||
|             try (Cursor c = getContentResolver().query(uri, projection, null, null, null)) { | ||||
|                 if (c != null && c.moveToFirst()) { | ||||
|                     final int fileNameColumnId = c.getColumnIndex(OpenableColumns.DISPLAY_NAME); | ||||
|                     if (fileNameColumnId >= 0) attachmentFileName = c.getString(fileNameColumnId); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (attachmentFileName == null) attachmentFileName = subjectFromIntent; | ||||
|  | ||||
|             InputStream in = getContentResolver().openInputStream(uri); | ||||
|             promptNameAndSave(in, attachmentFileName); | ||||
|         } catch (Exception e) { | ||||
|             showErrorDialogAndQuit("Unable to handle shared content:\n\n" + e.getMessage()); | ||||
|             Log.e("termux", "handleContentUri(uri=" + uri + ") failed", e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void promptNameAndSave(final InputStream in, final String attachmentFileName) { | ||||
|         DialogUtils.textInput(this, R.string.file_received_title, attachmentFileName | ||||
|             , android.R.string.ok, new DialogUtils.TextSetListener() { | ||||
|             @Override | ||||
|             public void onTextSet(final String text) { | ||||
|                 if (saveStreamWithName(in, text) == null) return; | ||||
|                 finish(); | ||||
|             } | ||||
|         }, R.string.file_received_open_folder_button, new DialogUtils.TextSetListener() { | ||||
|             @Override | ||||
|             public void onTextSet(String text) { | ||||
|                 if (saveStreamWithName(in, text) == null) return; | ||||
|  | ||||
|                 Intent executeIntent = new Intent(TermuxService.ACTION_EXECUTE); | ||||
|                 executeIntent.putExtra(TermuxService.EXTRA_CURRENT_WORKING_DIRECTORY, TERMUX_RECEIVEDIR); | ||||
|                 executeIntent.setClass(TermuxFileReceiverActivity.this, TermuxService.class); | ||||
|                 startService(executeIntent); | ||||
|                 finish(); | ||||
|             } | ||||
|         }, R.string.file_received_edit_button, new DialogUtils.TextSetListener() { | ||||
|             @Override | ||||
|             public void onTextSet(String text) { | ||||
|                 File outFile = saveStreamWithName(in, text); | ||||
|                 if (outFile == null) return; | ||||
|  | ||||
|                 final File editorProgramFile = new File(EDITOR_PROGRAM); | ||||
|                 if (!editorProgramFile.isFile()) { | ||||
|                     showErrorDialogAndQuit("The following file does not exist:\n$HOME/bin/termux-file-editor\n\n" | ||||
|                         + "Create this file as a script or a symlink - it will be called with the received file as only argument."); | ||||
|                     return; | ||||
|                 } | ||||
|  | ||||
|                 // Do this for the user if necessary: | ||||
|                 editorProgramFile.setExecutable(true); | ||||
|  | ||||
|                 final Uri scriptUri = new Uri.Builder().scheme("file").path(EDITOR_PROGRAM).build(); | ||||
|  | ||||
|                 Intent executeIntent = new Intent(TermuxService.ACTION_EXECUTE, scriptUri); | ||||
|                 executeIntent.setClass(TermuxFileReceiverActivity.this, TermuxService.class); | ||||
|                 executeIntent.putExtra(TermuxService.EXTRA_ARGUMENTS, new String[]{outFile.getAbsolutePath()}); | ||||
|                 startService(executeIntent); | ||||
|                 finish(); | ||||
|             } | ||||
|         }, new DialogInterface.OnDismissListener() { | ||||
|             @Override | ||||
|             public void onDismiss(DialogInterface dialog) { | ||||
|                 if (mFinishOnDismissNameDialog) finish(); | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     public File saveStreamWithName(InputStream in, String attachmentFileName) { | ||||
|         File receiveDir = new File(TERMUX_RECEIVEDIR); | ||||
|         if (!receiveDir.isDirectory() && !receiveDir.mkdirs()) { | ||||
|             showErrorDialogAndQuit("Cannot create directory: " + receiveDir.getAbsolutePath()); | ||||
|             return null; | ||||
|         } | ||||
|         try { | ||||
|             final File outFile = new File(receiveDir, attachmentFileName); | ||||
|             try (FileOutputStream f = new FileOutputStream(outFile)) { | ||||
|                 byte[] buffer = new byte[4096]; | ||||
|                 int readBytes; | ||||
|                 while ((readBytes = in.read(buffer)) > 0) { | ||||
|                     f.write(buffer, 0, readBytes); | ||||
|                 } | ||||
|             } | ||||
|             return outFile; | ||||
|         } catch (IOException e) { | ||||
|             showErrorDialogAndQuit("Error saving file:\n\n" + e); | ||||
|             Log.e("termux", "Error saving file", e); | ||||
|             return null; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void handleUrlAndFinish(final String url) { | ||||
|         final File urlOpenerProgramFile = new File(URL_OPENER_PROGRAM); | ||||
|         if (!urlOpenerProgramFile.isFile()) { | ||||
|             showErrorDialogAndQuit("The following file does not exist:\n$HOME/bin/termux-url-opener\n\n" | ||||
|                 + "Create this file as a script or a symlink - it will be called with the shared URL as only argument."); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Do this for the user if necessary: | ||||
|         urlOpenerProgramFile.setExecutable(true); | ||||
|  | ||||
|         final Uri urlOpenerProgramUri = new Uri.Builder().scheme("file").path(URL_OPENER_PROGRAM).build(); | ||||
|  | ||||
|         Intent executeIntent = new Intent(TermuxService.ACTION_EXECUTE, urlOpenerProgramUri); | ||||
|         executeIntent.setClass(TermuxFileReceiverActivity.this, TermuxService.class); | ||||
|         executeIntent.putExtra(TermuxService.EXTRA_ARGUMENTS, new String[]{url}); | ||||
|         startService(executeIntent); | ||||
|         finish(); | ||||
|     } | ||||
|  | ||||
| } | ||||
		Reference in New Issue
	
	Block a user