package eu.faircode.email; /* This file is part of FairEmail. FairEmail 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. FairEmail 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 FairEmail. If not, see . Copyright 2018-2019 by Marcel Bokhorst (M66B) */ import android.Manifest; import android.content.ContentResolver; import android.content.Context; import android.content.DialogInterface; import android.content.Intent; import android.content.res.AssetFileDescriptor; import android.net.Uri; import android.os.Bundle; import android.os.ParcelFileDescriptor; import android.text.Spanned; import android.text.TextUtils; import android.view.Menu; import android.view.MenuInflater; import android.view.MenuItem; import android.view.View; import android.widget.ArrayAdapter; import android.widget.TextView; import android.widget.Toast; import androidx.annotation.NonNull; import androidx.annotation.Nullable; import androidx.appcompat.app.AlertDialog; import androidx.constraintlayout.widget.Group; import androidx.recyclerview.widget.LinearLayoutManager; import androidx.recyclerview.widget.RecyclerView; import com.google.android.material.snackbar.Snackbar; import com.sun.mail.imap.IMAPFolder; import org.jsoup.nodes.Document; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.InputStream; import java.io.OutputStream; import java.util.List; import java.util.Properties; import javax.mail.Flags; import javax.mail.Folder; import javax.mail.Message; import javax.mail.Session; import javax.mail.internet.MimeMessage; public class ActivityEML extends ActivityBase { private Uri uri; private MessageHelper.AttachmentPart apart; private static final int REQUEST_ATTACHMENT = 1; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); getSupportActionBar().setSubtitle("EML"); setContentView(R.layout.activity_eml); final TextView tvTo = findViewById(R.id.tvTo); final TextView tvFrom = findViewById(R.id.tvFrom); final TextView tvSubject = findViewById(R.id.tvSubject); final View vSeparatorAttachments = findViewById(R.id.vSeparatorAttachments); final RecyclerView rvAttachment = findViewById(R.id.rvAttachment); final TextView tvBody = findViewById(R.id.tvBody); final ContentLoadingProgressBar pbWait = findViewById(R.id.pbWait); final Group grpReady = findViewById(R.id.grpReady); rvAttachment.setHasFixedSize(false); LinearLayoutManager llm = new LinearLayoutManager(this); rvAttachment.setLayoutManager(llm); vSeparatorAttachments.setVisibility(View.GONE); grpReady.setVisibility(View.GONE); uri = getIntent().getData(); if (uri == null) { pbWait.setVisibility(View.GONE); return; } else pbWait.setVisibility(View.VISIBLE); Log.i("EML uri=" + uri); Bundle args = new Bundle(); args.putParcelable("uri", uri); new SimpleTask() { @Override protected void onPostExecute(Bundle args) { pbWait.setVisibility(View.GONE); } @Override protected Result onExecute(Context context, Bundle args) throws Throwable { Uri uri = args.getParcelable("uri"); if (!"content".equals(uri.getScheme()) && !Helper.hasPermission(context, Manifest.permission.READ_EXTERNAL_STORAGE)) { Log.w("EML uri=" + uri); throw new IllegalArgumentException(context.getString(R.string.title_no_stream)); } Result result = new Result(); ContentResolver resolver = context.getContentResolver(); AssetFileDescriptor descriptor = resolver.openTypedAssetFileDescriptor(uri, "*/*", null); try (InputStream is = descriptor.createInputStream()) { Properties props = MessageHelper.getSessionProperties(); Session isession = Session.getInstance(props, null); MimeMessage imessage = new MimeMessage(isession, is); MessageHelper helper = new MessageHelper(imessage); result.from = MessageHelper.formatAddresses(helper.getFrom()); result.to = MessageHelper.formatAddresses(helper.getTo()); result.subject = helper.getSubject(); result.parts = helper.getMessageParts(context); String html = result.parts.getHtml(context); if (html != null) { Document document = HtmlHelper.sanitize(context, html, false, false); result.body = HtmlHelper.fromHtml(document.html()); } return result; } } @Override protected void onExecuted(Bundle args, Result result) { tvFrom.setText(result.from); tvTo.setText(result.to); tvSubject.setText(result.subject); vSeparatorAttachments.setVisibility(result.parts.getAttachmentParts().size() > 0 ? View.VISIBLE : View.GONE); AdapterAttachmentEML adapter = new AdapterAttachmentEML( ActivityEML.this, result.parts.getAttachmentParts(), new AdapterAttachmentEML.IEML() { @Override public void onSelected(MessageHelper.AttachmentPart apart) { ActivityEML.this.apart = apart; Intent create = new Intent(Intent.ACTION_CREATE_DOCUMENT); create.addCategory(Intent.CATEGORY_OPENABLE); create.setType(apart.attachment.getMimeType()); if (!TextUtils.isEmpty(apart.attachment.name)) create.putExtra(Intent.EXTRA_TITLE, apart.attachment.name); if (create.resolveActivity(getPackageManager()) == null) ToastEx.makeText(ActivityEML.this, R.string.title_no_saf, Toast.LENGTH_LONG).show(); else startActivityForResult(Helper.getChooser(ActivityEML.this, create), REQUEST_ATTACHMENT); } }); rvAttachment.setAdapter(adapter); tvBody.setText(result.body); grpReady.setVisibility(View.VISIBLE); } @Override protected void onException(Bundle args, @NonNull Throwable ex) { if (ex instanceof IllegalArgumentException) Snackbar.make(findViewById(android.R.id.content), ex.getMessage(), Snackbar.LENGTH_LONG).show(); else Helper.unexpectedError(getSupportFragmentManager(), ex); } }.execute(this, args, "eml:decode"); } @Override protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) { super.onActivityResult(requestCode, resultCode, data); try { switch (requestCode) { case REQUEST_ATTACHMENT: if (resultCode == RESULT_OK && data != null) onSaveAttachment(data); break; } } catch (Throwable ex) { Log.e(ex); } } private void onSaveAttachment(Intent data) { Bundle args = new Bundle(); args.putParcelable("uri", data.getData()); new SimpleTask() { @Override protected Void onExecute(Context context, Bundle args) throws Throwable { Uri uri = args.getParcelable("uri"); if (!"content".equals(uri.getScheme())) { Log.w("Save attachment uri=" + uri); throw new IllegalArgumentException(context.getString(R.string.title_no_stream)); } ParcelFileDescriptor pfd = null; OutputStream os = null; InputStream is; try { pfd = getContentResolver().openFileDescriptor(uri, "w"); os = new FileOutputStream(pfd.getFileDescriptor()); is = apart.part.getInputStream(); byte[] buffer = new byte[Helper.BUFFER_SIZE]; int read; while ((read = is.read(buffer)) != -1) os.write(buffer, 0, read); } finally { try { if (pfd != null) pfd.close(); } catch (Throwable ex) { Log.w(ex); } try { if (os != null) os.close(); } catch (Throwable ex) { Log.w(ex); } } return null; } @Override protected void onExecuted(Bundle args, Void data) { ToastEx.makeText(ActivityEML.this, R.string.title_attachment_saved, Toast.LENGTH_LONG).show(); } @Override protected void onException(Bundle args, Throwable ex) { if (ex instanceof IllegalArgumentException || ex instanceof FileNotFoundException) ToastEx.makeText(ActivityEML.this, ex.getMessage(), Toast.LENGTH_LONG).show(); else Helper.unexpectedError(getSupportFragmentManager(), ex); } }.execute(this, args, "eml:attachment"); } @Override public boolean onCreateOptionsMenu(Menu menu) { MenuInflater inflater = getMenuInflater(); inflater.inflate(R.menu.menu_eml, menu); return true; } @Override public boolean onOptionsItemSelected(@NonNull MenuItem item) { switch (item.getItemId()) { case R.id.menu_save: onMenuSave(); return true; default: return super.onOptionsItemSelected(item); } } private void onMenuSave() { new SimpleTask>() { @Override protected List onExecute(Context context, Bundle args) { DB db = DB.getInstance(context); return db.account().getSynchronizingAccounts(); } @Override protected void onExecuted(Bundle args, List accounts) { ArrayAdapter adapter = new ArrayAdapter<>(ActivityEML.this, R.layout.spinner_item1, android.R.id.text1); adapter.addAll(accounts); new AlertDialog.Builder(ActivityEML.this) .setTitle(R.string.title_save_eml) .setAdapter(adapter, new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { EntityAccount account = adapter.getItem(which); Bundle args = new Bundle(); args.putParcelable("uri", uri); args.putLong("account", account.id); new SimpleTask() { @Override protected void onPreExecute(Bundle args) { ToastEx.makeText(ActivityEML.this, R.string.title_executing, Toast.LENGTH_LONG).show(); } @Override protected String onExecute(Context context, Bundle args) throws Throwable { Uri uri = args.getParcelable("uri"); long aid = args.getLong("account"); DB db = DB.getInstance(context); EntityAccount account = db.account().getAccount(aid); if (account == null) return null; EntityFolder inbox = db.folder().getFolderByType(account.id, EntityFolder.INBOX); if (inbox == null) throw new IllegalArgumentException(context.getString(R.string.title_no_folder)); ContentResolver resolver = context.getContentResolver(); AssetFileDescriptor descriptor = resolver.openTypedAssetFileDescriptor(uri, "*/*", null); try (InputStream is = descriptor.createInputStream()) { Properties props = MessageHelper.getSessionProperties(); Session isession = Session.getInstance(props, null); MimeMessage imessage = new MimeMessage(isession, is); try (MailService iservice = new MailService(context, account.getProtocol(), account.realm, account.insecure, false, true)) { iservice.setPartialFetch(account.partial_fetch); iservice.setIgnoreBodyStructureSize(account.ignore_size); iservice.setSeparateStoreConnection(); iservice.connect(account); IMAPFolder ifolder = (IMAPFolder) iservice.getStore().getFolder(inbox.name); ifolder.open(Folder.READ_WRITE); if (ifolder.getPermanentFlags().contains(Flags.Flag.DRAFT)) imessage.setFlag(Flags.Flag.DRAFT, false); ifolder.appendMessages(new Message[]{imessage}); EntityOperation.sync(context, inbox.id, true); } } return account.name + "/" + inbox.name; } @Override protected void onExecuted(Bundle args, String name) { ToastEx.makeText(ActivityEML.this, name, Toast.LENGTH_LONG).show(); } @Override protected void onException(Bundle args, @NonNull Throwable ex) { if (ex instanceof IllegalArgumentException) Snackbar.make(findViewById(android.R.id.content), ex.getMessage(), Snackbar.LENGTH_LONG).show(); else Helper.unexpectedError(getSupportFragmentManager(), ex); } }.execute(ActivityEML.this, args, "eml:store"); } }) .setNegativeButton(android.R.string.cancel, null) .show(); } @Override protected void onException(Bundle args, @NonNull Throwable ex) { Helper.unexpectedError(getSupportFragmentManager(), ex); } }.execute(this, new Bundle(), "messages:accounts"); } private class Result { String from; String to; String subject; MessageHelper.MessageParts parts; Spanned body; } }