package com.securecomcode.text; import android.app.Activity; import android.app.AlertDialog; import android.app.ProgressDialog; import android.content.Context; import android.content.DialogInterface; import android.content.Intent; import android.database.Cursor; import android.media.MediaScannerConnection; import android.os.AsyncTask; import android.os.Bundle; import android.os.Environment; import android.os.Handler; import android.support.v4.app.LoaderManager; import android.support.v4.content.Loader; import android.text.ClipboardManager; import android.util.Log; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.support.v4.widget.CursorAdapter; import android.webkit.MimeTypeMap; import android.widget.AdapterView; import android.widget.ListView; import android.widget.Toast; import com.actionbarsherlock.app.SherlockListFragment; import com.actionbarsherlock.view.ActionMode; import com.actionbarsherlock.view.Menu; import com.actionbarsherlock.view.MenuInflater; import com.actionbarsherlock.view.MenuItem; import com.securecomcode.text.database.DatabaseFactory; import com.securecomcode.text.database.loaders.ConversationLoader; import com.securecomcode.text.database.model.MediaMmsMessageRecord; import com.securecomcode.text.database.model.MessageRecord; import com.securecomcode.text.mms.Slide; import com.securecomcode.text.recipients.RecipientFactory; import com.securecomcode.text.recipients.Recipients; import com.securecomcode.text.sms.MessageSender; import com.securecomcode.text.util.Dialogs; import com.securecomcode.text.util.DirectoryHelper; import org.whispersystems.textsecure.crypto.MasterSecret; import org.whispersystems.textsecure.util.Util; import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.lang.ref.WeakReference; import java.sql.Date; import java.text.SimpleDateFormat; import java.util.List; import java.util.concurrent.ExecutionException; public class ConversationFragment extends SherlockListFragment implements LoaderManager.LoaderCallbacks<Cursor> { private static final String TAG = ConversationFragment.class.getSimpleName(); private ConversationFragmentListener listener; private MasterSecret masterSecret; private Recipients recipients; private long threadId; private ActionMode actionMode; @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle bundle) { return inflater.inflate(R.layout.conversation_fragment, container, false); } @Override public void onActivityCreated(Bundle bundle) { super.onActivityCreated(bundle); initializeResources(); initializeListAdapter(); initializeContextualActionBar(); } @Override public void onAttach(Activity activity) { super.onAttach(activity); this.listener = (ConversationFragmentListener)activity; } private void initializeResources() { String recipientIds = this.getActivity().getIntent().getStringExtra("recipients"); this.masterSecret = this.getActivity().getIntent().getParcelableExtra("master_secret"); this.recipients = RecipientFactory.getRecipientsForIds(getActivity(), recipientIds, true); this.threadId = this.getActivity().getIntent().getLongExtra("thread_id", -1); } private void initializeListAdapter() { if (this.recipients != null && this.threadId != -1) { this.setListAdapter(new ConversationAdapter(getActivity(), masterSecret, new FailedIconClickHandler(), (!this.recipients.isSingleRecipient()) || this.recipients.isGroupRecipient(), DirectoryHelper.isPushDestination(getActivity(), this.recipients))); getListView().setRecyclerListener((ConversationAdapter)getListAdapter()); getLoaderManager().initLoader(0, null, this); } } private void initializeContextualActionBar() { getListView().setOnItemLongClickListener(new AdapterView.OnItemLongClickListener() { @Override public boolean onItemLongClick(AdapterView<?> parent, View view, int position, long id) { if (actionMode != null) { view.setSelected(true); return false; } actionMode = getSherlockActivity().startActionMode(actionModeCallback); view.setSelected(true); return true; } }); getListView().setOnItemClickListener(new AdapterView.OnItemClickListener() { @Override public void onItemClick(AdapterView<?> parent, View view, int position, long id) { if (actionMode != null) { view.setSelected(true); setCorrectMenuVisibility(getMessageRecord(), actionMode.getMenu()); } } }); } private void setCorrectMenuVisibility(MessageRecord messageRecord, Menu menu) { MenuItem resend = menu.findItem(R.id.menu_context_resend); MenuItem saveAttachment = menu.findItem(R.id.menu_context_save_attachment); if (messageRecord.isFailed()) resend.setVisible(true); else resend.setVisible(false); if (messageRecord.isMms() && !messageRecord.isMmsNotification()) { try { if (((MediaMmsMessageRecord)messageRecord).getSlideDeck().get().containsMediaSlide()) { saveAttachment.setVisible(true); } else { saveAttachment.setVisible(false); } } catch (InterruptedException ie) { Log.w(TAG, ie); } catch (ExecutionException ee) { Log.w(TAG, ee); } } else { saveAttachment.setVisible(false); } } private MessageRecord getMessageRecord() { Cursor cursor = ((CursorAdapter)getListAdapter()).getCursor(); ConversationItem conversationItem = (ConversationItem)(((ConversationAdapter)getListAdapter()).newView(getActivity(), cursor, null)); return conversationItem.getMessageRecord(); } public void reload(Recipients recipients, long threadId) { this.recipients = recipients; this.threadId = threadId; initializeListAdapter(); } public void scrollToBottom() { final ListView list = getListView(); list.post(new Runnable() { @Override public void run() { list.setSelection(getListAdapter().getCount() - 1); } }); } private void handleCopyMessage(MessageRecord message) { String body = message.getDisplayBody().toString(); if (body == null) return; ClipboardManager clipboard = (ClipboardManager)getActivity() .getSystemService(Context.CLIPBOARD_SERVICE); clipboard.setText(body); } private void handleDeleteMessage(final MessageRecord message) { final long messageId = message.getId(); AlertDialog.Builder builder = new AlertDialog.Builder(getActivity()); builder.setTitle(R.string.ConversationFragment_confirm_message_delete); builder.setIcon(Dialogs.resolveIcon(getActivity(), R.attr.dialog_alert_icon)); builder.setCancelable(true); builder.setMessage(R.string.ConversationFragment_are_you_sure_you_want_to_permanently_delete_this_message); builder.setPositiveButton(R.string.yes, new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { if (message.isMms()) { DatabaseFactory.getMmsDatabase(getActivity()).delete(messageId); } else { DatabaseFactory.getSmsDatabase(getActivity()).deleteMessage(messageId); } } }); builder.setNegativeButton(R.string.no, null); builder.show(); } private void handleDisplayDetails(MessageRecord message) { long dateReceived = message.getDateReceived(); long dateSent = message.getDateSent(); String transport; if (message.isPending()) transport = "pending"; else if (message.isPush()) transport = "push"; else if (message.isMms()) transport = "mms"; else transport = "sms"; SimpleDateFormat dateFormatter = new SimpleDateFormat("EEE MMM d, yyyy 'at' hh:mm:ss a zzz"); AlertDialog.Builder builder = new AlertDialog.Builder(getActivity()); builder.setTitle(R.string.ConversationFragment_message_details); builder.setIcon(Dialogs.resolveIcon(getActivity(), R.attr.dialog_info_icon)); builder.setCancelable(true); if (dateReceived == dateSent || message.isOutgoing()) { builder.setMessage(String.format(getSherlockActivity() .getString(R.string.ConversationFragment_transport_s_sent_received_s), transport.toUpperCase(), dateFormatter.format(new Date(dateSent)))); } else { builder.setMessage(String.format(getSherlockActivity() .getString(R.string.ConversationFragment_sender_s_transport_s_sent_s_received_s), message.getIndividualRecipient().getNumber(), transport.toUpperCase(), dateFormatter.format(new Date(dateSent)), dateFormatter.format(new Date(dateReceived)))); } builder.setPositiveButton(android.R.string.ok, null); builder.show(); } private void handleForwardMessage(MessageRecord message) { Intent composeIntent = new Intent(getActivity(), ShareActivity.class); composeIntent.putExtra(ConversationActivity.DRAFT_TEXT_EXTRA, message.getDisplayBody().toString()); composeIntent.putExtra(ShareActivity.MASTER_SECRET_EXTRA, masterSecret); startActivity(composeIntent); } private void handleResendMessage(MessageRecord message) { long messageId = message.getId(); final Activity activity = getActivity(); MessageSender.resend(activity, messageId, message.isMms()); } private void handleSaveAttachment(final MessageRecord message) { AlertDialog.Builder builder = new AlertDialog.Builder(getActivity()); builder.setTitle(R.string.ConversationFragment_save_to_sd_card); builder.setIcon(Dialogs.resolveIcon(getActivity(), R.attr.dialog_alert_icon)); builder.setCancelable(true); builder.setMessage(R.string.ConversationFragment_this_media_has_been_stored_in_an_encrypted_database_warning); builder.setPositiveButton(R.string.yes, new DialogInterface.OnClickListener() { public void onClick(DialogInterface dialog, int which) { SaveAttachmentTask saveTask = new SaveAttachmentTask(getActivity()); saveTask.execute((MediaMmsMessageRecord) message); } }); builder.setNegativeButton(R.string.no, null); builder.show(); } @Override public Loader<Cursor> onCreateLoader(int arg0, Bundle arg1) { return new ConversationLoader(getActivity(), threadId); } @Override public void onLoadFinished(Loader<Cursor> arg0, Cursor cursor) { ((CursorAdapter)getListAdapter()).changeCursor(cursor); } @Override public void onLoaderReset(Loader<Cursor> arg0) { ((CursorAdapter)getListAdapter()).changeCursor(null); } private class FailedIconClickHandler extends Handler { @Override public void handleMessage(android.os.Message message) { if (listener != null) { listener.setComposeText((String)message.obj); } } } public interface ConversationFragmentListener { public void setComposeText(String text); } private ActionMode.Callback actionModeCallback = new ActionMode.Callback() { @Override public boolean onCreateActionMode(ActionMode mode, Menu menu) { MenuInflater inflater = mode.getMenuInflater(); inflater.inflate(R.menu.conversation_context, menu); MessageRecord messageRecord = getMessageRecord(); setCorrectMenuVisibility(messageRecord, menu); return true; } @Override public boolean onPrepareActionMode(ActionMode actionMode, Menu menu) { return false; } @Override public void onDestroyActionMode(ActionMode mode) { if (getListView() != null && getListView().getChildCount() > 0) { for (int i = 0; i < getListView().getChildCount(); i++){ getListView().getChildAt(i).setSelected(false); } } actionMode = null; } @Override public boolean onActionItemClicked(ActionMode mode, MenuItem item) { MessageRecord messageRecord = getMessageRecord(); switch(item.getItemId()) { case R.id.menu_context_copy: handleCopyMessage(messageRecord); actionMode.finish(); return true; case R.id.menu_context_delete_message: handleDeleteMessage(messageRecord); actionMode.finish(); return true; case R.id.menu_context_details: handleDisplayDetails(messageRecord); actionMode.finish(); return true; case R.id.menu_context_forward: handleForwardMessage(messageRecord); actionMode.finish(); return true; case R.id.menu_context_resend: handleResendMessage(messageRecord); actionMode.finish(); return true; case R.id.menu_context_save_attachment: handleSaveAttachment(messageRecord); actionMode.finish(); return true; } return false; } }; private class SaveAttachmentTask extends AsyncTask<MediaMmsMessageRecord, Void, Integer> { private static final int SUCCESS = 0; private static final int FAILURE = 1; private static final int WRITE_ACCESS_FAILURE = 2; private final WeakReference<Context> contextReference; private ProgressDialog progressDialog; public SaveAttachmentTask(Context context) { this.contextReference = new WeakReference<Context>(context); } @Override protected void onPreExecute() { Context context = contextReference.get(); if (context != null) { progressDialog = ProgressDialog.show(context, context.getString(R.string.ConversationFragment_saving_attachment), context.getString(R.string.ConversationFragment_saving_attachment_to_sd_card), true, false); } } @Override protected Integer doInBackground(MediaMmsMessageRecord... messageRecord) { try { Context context = contextReference.get(); if (!Environment.getExternalStorageDirectory().canWrite()) { return WRITE_ACCESS_FAILURE; } if (context == null) { return FAILURE; } Slide slide = getAttachment(messageRecord[0]); if (slide == null) { return FAILURE; } File mediaFile = constructOutputFile(slide, messageRecord[0].getDateReceived()); InputStream inputStream = slide.getPartDataInputStream(); OutputStream outputStream = new FileOutputStream(mediaFile); Util.copy(inputStream, outputStream); MediaScannerConnection.scanFile(context, new String[] {mediaFile.getAbsolutePath()}, new String[] {slide.getContentType()}, null); return SUCCESS; } catch (IOException ioe) { Log.w(TAG, ioe); return FAILURE; } catch (InterruptedException e) { throw new AssertionError(e); } catch (ExecutionException e) { Log.w(TAG, e); return FAILURE; } } @Override protected void onPostExecute(Integer result) { Context context = contextReference.get(); if (context == null) return; switch (result) { case FAILURE: Toast.makeText(context, R.string.ConversationFragment_error_while_saving_attachment_to_sd_card, Toast.LENGTH_LONG).show(); break; case SUCCESS: Toast.makeText(context, R.string.ConversationFragment_success_exclamation, Toast.LENGTH_LONG).show(); break; case WRITE_ACCESS_FAILURE: Toast.makeText(context, R.string.ConversationFragment_unable_to_write_to_sd_card_exclamation, Toast.LENGTH_LONG).show(); break; } if (progressDialog != null) progressDialog.dismiss(); } private Slide getAttachment(MediaMmsMessageRecord record) throws ExecutionException, InterruptedException { List<Slide> slides = record.getSlideDeck().get().getSlides(); for (Slide slide : slides) { if (slide.hasImage() || slide.hasVideo() || slide.hasAudio()) { return slide; } } return null; } private File constructOutputFile(Slide slide, long timestamp) throws IOException { File sdCard = Environment.getExternalStorageDirectory(); File outputDirectory; if (slide.hasVideo()) { outputDirectory = new File(sdCard.getAbsoluteFile() + File.separator + "Movies"); } else if (slide.hasAudio()) { outputDirectory = new File(sdCard.getAbsolutePath() + File.separator + "Music"); } else { outputDirectory = new File(sdCard.getAbsolutePath() + File.separator + "Pictures"); } outputDirectory.mkdirs(); MimeTypeMap mimeTypeMap = MimeTypeMap.getSingleton(); String extension = mimeTypeMap.getExtensionFromMimeType(slide.getContentType()); SimpleDateFormat dateFormatter = new SimpleDateFormat("yyyy-MM-dd-HHmmss"); String base = "textsecure-" + dateFormatter.format(timestamp); if (extension == null) extension = "attach"; int i = 0; File file = new File(outputDirectory, base + "." + extension); while (file.exists()) { file = new File(outputDirectory, base + "-" + (++i) + "." + extension); } return file; } } }