Forward as Attachment

This commit is contained in:
Thomas E. Horner 2016-12-13 17:31:59 +01:00 committed by cketti
parent d0c8cc3b57
commit 4f4e1386cb
11 changed files with 136 additions and 43 deletions

View file

@ -1,6 +1,7 @@
package com.fsck.k9.activity; package com.fsck.k9.activity;
import java.io.IOException;
import java.util.Date; import java.util.Date;
import java.util.HashMap; import java.util.HashMap;
import java.util.List; import java.util.List;
@ -121,6 +122,7 @@ public class MessageCompose extends K9Activity implements OnClickListener,
public static final String ACTION_REPLY = "com.fsck.k9.intent.action.REPLY"; public static final String ACTION_REPLY = "com.fsck.k9.intent.action.REPLY";
public static final String ACTION_REPLY_ALL = "com.fsck.k9.intent.action.REPLY_ALL"; public static final String ACTION_REPLY_ALL = "com.fsck.k9.intent.action.REPLY_ALL";
public static final String ACTION_FORWARD = "com.fsck.k9.intent.action.FORWARD"; public static final String ACTION_FORWARD = "com.fsck.k9.intent.action.FORWARD";
public static final String ACTION_FORWARD_AS_ATTACHMENT = "com.fsck.k9.intent.action.FORWARD_AS_ATTACHMENT";
public static final String ACTION_EDIT_DRAFT = "com.fsck.k9.intent.action.EDIT_DRAFT"; public static final String ACTION_EDIT_DRAFT = "com.fsck.k9.intent.action.EDIT_DRAFT";
private static final String ACTION_AUTOCRYPT_PEER = "org.autocrypt.PEER_ACTION"; private static final String ACTION_AUTOCRYPT_PEER = "org.autocrypt.PEER_ACTION";
@ -355,6 +357,8 @@ public class MessageCompose extends K9Activity implements OnClickListener,
this.action = Action.REPLY_ALL; this.action = Action.REPLY_ALL;
} else if (ACTION_FORWARD.equals(action)) { } else if (ACTION_FORWARD.equals(action)) {
this.action = Action.FORWARD; this.action = Action.FORWARD;
} else if (ACTION_FORWARD_AS_ATTACHMENT.equals(action)) {
this.action = Action.FORWARD_AS_ATTACHMENT;
} else if (ACTION_EDIT_DRAFT.equals(action)) { } else if (ACTION_EDIT_DRAFT.equals(action)) {
this.action = Action.EDIT_DRAFT; this.action = Action.EDIT_DRAFT;
} else { } else {
@ -388,7 +392,8 @@ public class MessageCompose extends K9Activity implements OnClickListener,
if (!relatedMessageProcessed) { if (!relatedMessageProcessed) {
if (action == Action.REPLY || action == Action.REPLY_ALL || if (action == Action.REPLY || action == Action.REPLY_ALL ||
action == Action.FORWARD || action == Action.EDIT_DRAFT) { action == Action.FORWARD || action == Action.FORWARD_AS_ATTACHMENT ||
action == Action.EDIT_DRAFT) {
messageLoaderHelper = new MessageLoaderHelper(this, getLoaderManager(), getFragmentManager(), messageLoaderHelper = new MessageLoaderHelper(this, getLoaderManager(), getFragmentManager(),
messageLoaderCallbacks); messageLoaderCallbacks);
internalMessageHandler.sendEmptyMessage(MSG_PROGRESS_ON); internalMessageHandler.sendEmptyMessage(MSG_PROGRESS_ON);
@ -418,7 +423,7 @@ public class MessageCompose extends K9Activity implements OnClickListener,
recipientMvpView.requestFocusOnToField(); recipientMvpView.requestFocusOnToField();
} }
if (action == Action.FORWARD) { if (action == Action.FORWARD || action == Action.FORWARD_AS_ATTACHMENT) {
relatedMessageReference = relatedMessageReference.withModifiedFlag(Flag.FORWARDED); relatedMessageReference = relatedMessageReference.withModifiedFlag(Flag.FORWARDED);
} }
@ -1193,7 +1198,11 @@ public class MessageCompose extends K9Activity implements OnClickListener,
break; break;
} }
case FORWARD: { case FORWARD: {
processMessageToForward(messageViewInfo); processMessageToForward(messageViewInfo, false);
break;
}
case FORWARD_AS_ATTACHMENT: {
processMessageToForward(messageViewInfo, true);
break; break;
} }
case EDIT_DRAFT: { case EDIT_DRAFT: {
@ -1205,12 +1214,12 @@ public class MessageCompose extends K9Activity implements OnClickListener,
break; break;
} }
} }
} catch (MessagingException me) { } catch (MessagingException | IOException e) {
/* /*
* Let the user continue composing their message even if we have a problem processing * Let the user continue composing their message even if we have a problem processing
* the source message. Log it as an error, though. * the source message. Log it as an error, though.
*/ */
Timber.e(me, "Error while processing source message: "); Timber.e(e, "Error while processing source message: ");
} finally { } finally {
relatedMessageProcessed = true; relatedMessageProcessed = true;
changesMadeSinceLastSave = false; changesMadeSinceLastSave = false;
@ -1268,7 +1277,7 @@ public class MessageCompose extends K9Activity implements OnClickListener,
} }
private void processMessageToForward(MessageViewInfo messageViewInfo) throws MessagingException { private void processMessageToForward(MessageViewInfo messageViewInfo, boolean asAttachment) throws IOException, MessagingException {
Message message = messageViewInfo.message; Message message = messageViewInfo.message;
String subject = message.getSubject(); String subject = message.getSubject();
@ -1290,9 +1299,13 @@ public class MessageCompose extends K9Activity implements OnClickListener,
} }
// Quote the message and setup the UI. // Quote the message and setup the UI.
if (asAttachment) {
attachmentPresenter.processMessageToForwardAsAttachment(messageViewInfo);
} else {
quotedMessagePresenter.processMessageToForward(messageViewInfo); quotedMessagePresenter.processMessageToForward(messageViewInfo);
attachmentPresenter.processMessageToForward(messageViewInfo); attachmentPresenter.processMessageToForward(messageViewInfo);
} }
}
private void processDraftMessage(MessageViewInfo messageViewInfo) { private void processDraftMessage(MessageViewInfo messageViewInfo) {
Message message = messageViewInfo.message; Message message = messageViewInfo.message;
@ -1772,6 +1785,12 @@ public class MessageCompose extends K9Activity implements OnClickListener,
Toast.makeText(MessageCompose.this, Toast.makeText(MessageCompose.this,
getString(R.string.message_compose_attachments_skipped_toast), Toast.LENGTH_LONG).show(); getString(R.string.message_compose_attachments_skipped_toast), Toast.LENGTH_LONG).show();
} }
@Override
public void showMissingAttachmentsPartialMessageForwardWarning() {
Toast.makeText(MessageCompose.this,
getString(R.string.message_compose_attachments_forward_toast), Toast.LENGTH_LONG).show();
}
}; };
private Handler internalMessageHandler = new Handler() { private Handler internalMessageHandler = new Handler() {
@ -1809,6 +1828,7 @@ public class MessageCompose extends K9Activity implements OnClickListener,
REPLY(R.string.compose_title_reply), REPLY(R.string.compose_title_reply),
REPLY_ALL(R.string.compose_title_reply_all), REPLY_ALL(R.string.compose_title_reply_all),
FORWARD(R.string.compose_title_forward), FORWARD(R.string.compose_title_forward),
FORWARD_AS_ATTACHMENT(R.string.compose_title_forward_as_attachment),
EDIT_DRAFT(R.string.compose_title_compose); EDIT_DRAFT(R.string.compose_title_compose);
private final int titleResource; private final int titleResource;

View file

@ -895,6 +895,10 @@ public class MessageList extends K9Activity implements MessageListFragmentListen
mMessageViewFragment.onForward(); mMessageViewFragment.onForward();
return true; return true;
} }
case R.id.forward_as_attachment: {
mMessageViewFragment.onForwardAsAttachment();
return true;
}
case R.id.share: { case R.id.share: {
mMessageViewFragment.onSendAlternate(); mMessageViewFragment.onSendAlternate();
return true; return true;
@ -1245,6 +1249,16 @@ public class MessageList extends K9Activity implements MessageListFragmentListen
MessageActions.actionForward(this, messageReference, decryptionResultForReply); MessageActions.actionForward(this, messageReference, decryptionResultForReply);
} }
@Override
public void onForwardAsAttachment(MessageReference messageReference) {
onForwardAsAttachment(messageReference, null);
}
@Override
public void onForwardAsAttachment(MessageReference messageReference, Parcelable decryptionResultForReply) {
MessageActions.actionForwardAsAttachment(this, messageReference, decryptionResultForReply);
}
@Override @Override
public void onReply(MessageReference messageReference) { public void onReply(MessageReference messageReference) {
onReply(messageReference, null); onReply(messageReference, null);

View file

@ -1,6 +1,9 @@
package com.fsck.k9.activity.compose; package com.fsck.k9.activity.compose;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.LinkedHashMap; import java.util.LinkedHashMap;
@ -21,6 +24,7 @@ import com.fsck.k9.activity.loader.AttachmentContentLoader;
import com.fsck.k9.activity.loader.AttachmentInfoLoader; import com.fsck.k9.activity.loader.AttachmentInfoLoader;
import com.fsck.k9.activity.misc.Attachment; import com.fsck.k9.activity.misc.Attachment;
import com.fsck.k9.activity.misc.Attachment.LoadingState; import com.fsck.k9.activity.misc.Attachment.LoadingState;
import com.fsck.k9.mail.MessagingException;
import com.fsck.k9.mailstore.AttachmentViewInfo; import com.fsck.k9.mailstore.AttachmentViewInfo;
import com.fsck.k9.mailstore.MessageViewInfo; import com.fsck.k9.mailstore.MessageViewInfo;
@ -160,6 +164,10 @@ public class AttachmentPresenter {
} }
public boolean loadNonInlineAttachments(MessageViewInfo messageViewInfo) { public boolean loadNonInlineAttachments(MessageViewInfo messageViewInfo) {
return allPartsAvailable(messageViewInfo, true);
}
private boolean allPartsAvailable(MessageViewInfo messageViewInfo, boolean loadNonInlineAttachments) {
boolean allPartsAvailable = true; boolean allPartsAvailable = true;
for (AttachmentViewInfo attachmentViewInfo : messageViewInfo.attachments) { for (AttachmentViewInfo attachmentViewInfo : messageViewInfo.attachments) {
@ -170,8 +178,10 @@ public class AttachmentPresenter {
allPartsAvailable = false; allPartsAvailable = false;
continue; continue;
} }
if (loadNonInlineAttachments) {
addAttachment(attachmentViewInfo); addAttachment(attachmentViewInfo);
} }
}
return allPartsAvailable; return allPartsAvailable;
} }
@ -183,6 +193,27 @@ public class AttachmentPresenter {
} }
} }
public void processMessageToForwardAsAttachment(MessageViewInfo messageViewInfo) throws IOException, MessagingException {
boolean isMissingParts = !allPartsAvailable(messageViewInfo, false);
if (isMissingParts) {
attachmentMvpView.showMissingAttachmentsPartialMessageForwardWarning();
} else {
File tempFile = File.createTempFile("pre", ".tmp");
tempFile.deleteOnExit();
FileOutputStream fileOutputStream = new FileOutputStream(tempFile);
messageViewInfo.message.writeTo(fileOutputStream);
fileOutputStream.close();
int loaderId = getNextFreeLoaderId();
Attachment attachment = Attachment.createAttachment(null, loaderId, "message/rfc822")
.deriveWithMetadataLoaded("message/rfc822", messageViewInfo.message.getSubject(), tempFile.length())
.deriveWithLoadComplete(tempFile.getAbsolutePath());
attachments.put(attachment.uri, attachment);
attachmentMvpView.addAttachmentView(attachment);
}
}
private void addAttachmentAndStartLoader(Attachment attachment) { private void addAttachmentAndStartLoader(Attachment attachment) {
attachments.put(attachment.uri, attachment); attachments.put(attachment.uri, attachment);
listener.onAttachmentAdded(); listener.onAttachmentAdded();
@ -383,6 +414,7 @@ public class AttachmentPresenter {
void performSaveAfterChecks(); void performSaveAfterChecks();
void showMissingAttachmentsPartialMessageWarning(); void showMissingAttachmentsPartialMessageWarning();
void showMissingAttachmentsPartialMessageForwardWarning();
} }
public interface AttachmentsChangedListener { public interface AttachmentsChangedListener {

View file

@ -71,6 +71,17 @@ public class MessageActions {
context.startActivity(i); context.startActivity(i);
} }
/**
* Compose a new message as a forward of the given message.
*/
public static void actionForwardAsAttachment(Context context, MessageReference messageReference, Parcelable decryptionResult) {
Intent i = new Intent(context, MessageCompose.class);
i.putExtra(MessageCompose.EXTRA_MESSAGE_REFERENCE, messageReference.toIdentityString());
i.putExtra(MessageCompose.EXTRA_MESSAGE_DECRYPTION_RESULT, decryptionResult);
i.setAction(MessageCompose.ACTION_FORWARD_AS_ATTACHMENT);
context.startActivity(i);
}
/** /**
* Continue composition of the given message. This action modifies the way this Activity * Continue composition of the given message. This action modifies the way this Activity
* handles certain actions. * handles certain actions.

View file

@ -788,6 +788,10 @@ public class MessageListFragment extends Fragment implements OnItemClickListener
fragmentListener.onForward(messageReference); fragmentListener.onForward(messageReference);
} }
public void onForwardAsAttachment(MessageReference messageReference) {
fragmentListener.onForwardAsAttachment(messageReference);
}
private void onResendMessage(MessageReference messageReference) { private void onResendMessage(MessageReference messageReference) {
fragmentListener.onResendMessage(messageReference); fragmentListener.onResendMessage(messageReference);
} }
@ -1116,6 +1120,10 @@ public class MessageListFragment extends Fragment implements OnItemClickListener
onForward(getMessageAtPosition(adapterPosition)); onForward(getMessageAtPosition(adapterPosition));
break; break;
} }
case R.id.forward_as_attachment: {
onForwardAsAttachment(getMessageAtPosition(adapterPosition));
break;
}
case R.id.send_again: { case R.id.send_again: {
onResendMessage(getMessageAtPosition(adapterPosition)); onResendMessage(getMessageAtPosition(adapterPosition));
selectedCount = 0; selectedCount = 0;
@ -2376,6 +2384,7 @@ public class MessageListFragment extends Fragment implements OnItemClickListener
void showMoreFromSameSender(String senderAddress); void showMoreFromSameSender(String senderAddress);
void onResendMessage(MessageReference message); void onResendMessage(MessageReference message);
void onForward(MessageReference message); void onForward(MessageReference message);
void onForwardAsAttachment(MessageReference message);
void onReply(MessageReference message); void onReply(MessageReference message);
void onReplyAll(MessageReference message); void onReplyAll(MessageReference message);
void openMessage(MessageReference messageReference); void openMessage(MessageReference messageReference);

View file

@ -220,13 +220,6 @@ public abstract class MessageBuilder {
continue; continue;
} }
String contentType = attachment.contentType;
if (MimeUtil.isMessage(contentType)) {
contentType = "application/octet-stream";
// TODO reencode message body to 7 bit
// body = new TempFileMessageBody(attachment.filename);
}
Body body = new TempFileBody(attachment.filename); Body body = new TempFileBody(attachment.filename);
MimeBodyPart bp = new MimeBodyPart(body); MimeBodyPart bp = new MimeBodyPart(body);
@ -236,11 +229,12 @@ public abstract class MessageBuilder {
* MimeHeader.writeTo(). * MimeHeader.writeTo().
*/ */
bp.addHeader(MimeHeader.HEADER_CONTENT_TYPE, String.format("%s;\r\n name=\"%s\"", bp.addHeader(MimeHeader.HEADER_CONTENT_TYPE, String.format("%s;\r\n name=\"%s\"",
contentType, attachment.contentType,
EncoderUtil.encodeIfNecessary(attachment.name, EncoderUtil.encodeIfNecessary(attachment.name,
EncoderUtil.Usage.WORD_ENTITY, 7))); EncoderUtil.Usage.WORD_ENTITY, 7)));
bp.setEncoding(MimeUtility.getEncodingforType(contentType)); if (!MimeUtil.isMessage(attachment.contentType)) {
bp.setEncoding(MimeUtility.getEncodingforType(attachment.contentType));
/* /*
* TODO: Oh the joys of MIME... * TODO: Oh the joys of MIME...
@ -260,6 +254,7 @@ public abstract class MessageBuilder {
bp.addHeader(MimeHeader.HEADER_CONTENT_DISPOSITION, String.format(Locale.US, bp.addHeader(MimeHeader.HEADER_CONTENT_DISPOSITION, String.format(Locale.US,
"attachment;\r\n filename=\"%s\";\r\n size=%d", "attachment;\r\n filename=\"%s\";\r\n size=%d",
attachment.name, attachment.size)); attachment.name, attachment.size));
}
mp.addBodyPart(bp); mp.addBodyPart(bp);
} }

View file

@ -338,6 +338,12 @@ public class MessageViewFragment extends Fragment implements ConfirmationDialogF
} }
} }
public void onForwardAsAttachment() {
if (mMessage != null) {
mFragmentListener.onForwardAsAttachment(mMessage.makeMessageReference(), messageCryptoPresenter.getDecryptionResultForReply());
}
}
public void onToggleFlagged() { public void onToggleFlagged() {
if (mMessage != null) { if (mMessage != null) {
boolean newState = !mMessage.isSet(Flag.FLAGGED); boolean newState = !mMessage.isSet(Flag.FLAGGED);
@ -723,6 +729,7 @@ public class MessageViewFragment extends Fragment implements ConfirmationDialogF
public interface MessageViewFragmentListener { public interface MessageViewFragmentListener {
void onForward(MessageReference messageReference, Parcelable decryptionResultForReply); void onForward(MessageReference messageReference, Parcelable decryptionResultForReply);
void onForwardAsAttachment(MessageReference messageReference, Parcelable decryptionResultForReply);
void disableDeleteAction(); void disableDeleteAction();
void onReplyAll(MessageReference messageReference, Parcelable decryptionResultForReply); void onReplyAll(MessageReference messageReference, Parcelable decryptionResultForReply);
void onReply(MessageReference messageReference, Parcelable decryptionResultForReply); void onReply(MessageReference messageReference, Parcelable decryptionResultForReply);

View file

@ -17,6 +17,9 @@
<item <item
android:id="@+id/forward" android:id="@+id/forward"
android:title="@string/forward_action"/> android:title="@string/forward_action"/>
<item
android:id="@+id/forward_as_attachment"
android:title="@string/forward_as_attachment_action"/>
<item <item
android:id="@+id/delete" android:id="@+id/delete"
android:title="@string/delete_action" android:title="@string/delete_action"

View file

@ -97,6 +97,9 @@
<item <item
android:id="@+id/forward" android:id="@+id/forward"
android:title="@string/forward_action"/> android:title="@string/forward_action"/>
<item
android:id="@+id/forward_as_attachment"
android:title="@string/forward_as_attachment_action"/>
<item <item
android:id="@+id/share" android:id="@+id/share"
android:title="@string/send_alternate_action"/> android:title="@string/send_alternate_action"/>

View file

@ -97,6 +97,7 @@ Please submit bug reports, contribute new features and ask questions at
<string name="compose_title_reply">Reply</string> <string name="compose_title_reply">Reply</string>
<string name="compose_title_reply_all">Reply all</string> <string name="compose_title_reply_all">Reply all</string>
<string name="compose_title_forward">Forward</string> <string name="compose_title_forward">Forward</string>
<string name="compose_title_forward_as_attachment">Forward as attachment</string>
<string name="choose_account_title">Choose Account</string> <string name="choose_account_title">Choose Account</string>
<string name="choose_folder_title">Choose Folder</string> <string name="choose_folder_title">Choose Folder</string>
@ -131,6 +132,7 @@ Please submit bug reports, contribute new features and ask questions at
<string name="archive_action">Archive</string> <string name="archive_action">Archive</string>
<string name="spam_action">Spam</string> <string name="spam_action">Spam</string>
<string name="forward_action">Forward</string> <string name="forward_action">Forward</string>
<string name="forward_as_attachment_action">Forward as attachment</string>
<string name="move_action">Move</string> <string name="move_action">Move</string>
<string name="single_message_options_action">Send…</string> <string name="single_message_options_action">Send…</string>
<string name="refile_action">Refile…</string> <string name="refile_action">Refile…</string>
@ -276,6 +278,7 @@ Please submit bug reports, contribute new features and ask questions at
<string name="compose_error_incomplete_recipient">Recipient field contains incomplete input!</string> <string name="compose_error_incomplete_recipient">Recipient field contains incomplete input!</string>
<string name="error_contact_address_not_found">No email address could be found for this contact.</string> <string name="error_contact_address_not_found">No email address could be found for this contact.</string>
<string name="message_compose_attachments_skipped_toast">Some attachments cannot be forwarded because they have not been downloaded.</string> <string name="message_compose_attachments_skipped_toast">Some attachments cannot be forwarded because they have not been downloaded.</string>
<string name="message_compose_attachments_forward_toast">The message cannot be forwarded because some attachments have not been downloaded.</string>
<string name="message_compose_show_quoted_text_action">Quote message</string> <string name="message_compose_show_quoted_text_action">Quote message</string>
<string name="message_compose_description_delete_quoted_text">Remove quoted text</string> <string name="message_compose_description_delete_quoted_text">Remove quoted text</string>
<string name="message_compose_description_edit_quoted_text">Edit quoted text</string> <string name="message_compose_description_edit_quoted_text">Edit quoted text</string>

View file

@ -119,14 +119,10 @@ public class MessageBuilderTest {
"soviet message\r\n" + "soviet message\r\n" +
"text =E2=98=AD\r\n" + "text =E2=98=AD\r\n" +
"--" + BOUNDARY_1 + "\r\n" + "--" + BOUNDARY_1 + "\r\n" +
"Content-Type: application/octet-stream;\r\n" + "Content-Type: message/rfc822;\r\n" +
" name=\"attach.txt\"\r\n" + " name=\"attach.txt\"\r\n" +
"Content-Transfer-Encoding: base64\r\n" +
"Content-Disposition: attachment;\r\n" +
" filename=\"attach.txt\";\r\n" +
" size=23\r\n" +
"\r\n" + "\r\n" +
"dGV4dCBkYXRhIGluIGF0dGFjaG1lbnQ=\r\n" + "text data in attachment" +
"\r\n" + "\r\n" +
"--" + BOUNDARY_1 + "--\r\n"; "--" + BOUNDARY_1 + "--\r\n";
@ -193,7 +189,7 @@ public class MessageBuilderTest {
} }
@Test @Test
public void build_withMessageAttachment_shouldAttachAsApplicationOctetStream() throws Exception { public void build_withMessageAttachment_shouldAttachAsMessageRfc822() throws Exception {
MessageBuilder messageBuilder = createSimpleMessageBuilder(); MessageBuilder messageBuilder = createSimpleMessageBuilder();
Attachment attachment = createAttachmentWithContent("message/rfc822", "attach.txt", TEST_ATTACHMENT_TEXT); Attachment attachment = createAttachmentWithContent("message/rfc822", "attach.txt", TEST_ATTACHMENT_TEXT);
messageBuilder.setAttachments(Collections.singletonList(attachment)); messageBuilder.setAttachments(Collections.singletonList(attachment));