mirror of
https://github.com/oxen-io/session-android.git
synced 2025-01-07 23:47:43 +00:00
459 lines
19 KiB
Java
459 lines
19 KiB
Java
package org.thoughtcrime.securesms.mediasend;
|
|
|
|
import android.Manifest;
|
|
import android.arch.lifecycle.ViewModelProviders;
|
|
import android.content.Context;
|
|
import android.content.Intent;
|
|
import android.net.Uri;
|
|
import android.os.Bundle;
|
|
import android.support.annotation.NonNull;
|
|
import android.support.v4.app.Fragment;
|
|
import android.support.v4.app.FragmentManager;
|
|
import android.view.View;
|
|
import android.view.animation.AccelerateDecelerateInterpolator;
|
|
import android.view.animation.AccelerateInterpolator;
|
|
import android.view.animation.Animation;
|
|
import android.view.animation.DecelerateInterpolator;
|
|
import android.view.animation.OvershootInterpolator;
|
|
import android.view.animation.ScaleAnimation;
|
|
import android.widget.TextView;
|
|
import android.widget.Toast;
|
|
|
|
import org.thoughtcrime.securesms.PassphraseRequiredActionBarActivity;
|
|
import org.thoughtcrime.securesms.R;
|
|
import org.thoughtcrime.securesms.TransportOption;
|
|
import org.thoughtcrime.securesms.database.Address;
|
|
import org.thoughtcrime.securesms.logging.Log;
|
|
import org.thoughtcrime.securesms.permissions.Permissions;
|
|
import org.thoughtcrime.securesms.providers.BlobProvider;
|
|
import org.thoughtcrime.securesms.recipients.Recipient;
|
|
import org.thoughtcrime.securesms.scribbles.ScribbleFragment;
|
|
import org.thoughtcrime.securesms.util.DynamicLanguage;
|
|
import org.thoughtcrime.securesms.util.MediaUtil;
|
|
import org.thoughtcrime.securesms.util.Util;
|
|
import org.thoughtcrime.securesms.util.concurrent.SimpleTask;
|
|
import org.whispersystems.libsignal.util.guava.Optional;
|
|
|
|
import java.io.IOException;
|
|
import java.util.ArrayList;
|
|
import java.util.List;
|
|
import java.util.Locale;
|
|
|
|
/**
|
|
* Encompasses the entire flow of sending media, starting from the selection process to the actual
|
|
* captioning and editing of the content.
|
|
*
|
|
* This activity is intended to be launched via {@link #startActivityForResult(Intent, int)}.
|
|
* It will return the {@link Media} that the user decided to send.
|
|
*/
|
|
public class MediaSendActivity extends PassphraseRequiredActionBarActivity implements MediaPickerFolderFragment.Controller,
|
|
MediaPickerItemFragment.Controller,
|
|
MediaSendFragment.Controller,
|
|
ScribbleFragment.Controller,
|
|
Camera1Fragment.Controller
|
|
{
|
|
private static final String TAG = MediaSendActivity.class.getSimpleName();
|
|
|
|
public static final String EXTRA_MEDIA = "media";
|
|
public static final String EXTRA_MESSAGE = "message";
|
|
public static final String EXTRA_TRANSPORT = "transport";
|
|
|
|
|
|
private static final String KEY_ADDRESS = "address";
|
|
private static final String KEY_BODY = "body";
|
|
private static final String KEY_MEDIA = "media";
|
|
private static final String KEY_TRANSPORT = "transport";
|
|
private static final String KEY_IS_CAMERA = "is_camera";
|
|
|
|
private static final String TAG_FOLDER_PICKER = "folder_picker";
|
|
private static final String TAG_ITEM_PICKER = "item_picker";
|
|
private static final String TAG_SEND = "send";
|
|
private static final String TAG_CAMERA = "camera";
|
|
|
|
private final DynamicLanguage dynamicLanguage = new DynamicLanguage();
|
|
|
|
private Recipient recipient;
|
|
private TransportOption transport;
|
|
private MediaSendViewModel viewModel;
|
|
|
|
private View countButton;
|
|
private TextView countButtonText;
|
|
private View cameraButton;
|
|
|
|
/**
|
|
* Get an intent to launch the media send flow starting with the picker.
|
|
*/
|
|
public static Intent buildGalleryIntent(@NonNull Context context, @NonNull Recipient recipient, @NonNull String body, @NonNull TransportOption transport) {
|
|
Intent intent = new Intent(context, MediaSendActivity.class);
|
|
intent.putExtra(KEY_ADDRESS, recipient.getAddress().serialize());
|
|
intent.putExtra(KEY_TRANSPORT, transport);
|
|
intent.putExtra(KEY_BODY, body);
|
|
return intent;
|
|
}
|
|
|
|
/**
|
|
* Get an intent to launch the media send flow starting with the picker.
|
|
*/
|
|
public static Intent buildCameraIntent(@NonNull Context context, @NonNull Recipient recipient, @NonNull TransportOption transport) {
|
|
Intent intent = buildGalleryIntent(context, recipient, "", transport);
|
|
intent.putExtra(KEY_IS_CAMERA, true);
|
|
return intent;
|
|
}
|
|
|
|
/**
|
|
* Get an intent to launch the media send flow with a specific list of media. Will jump right to
|
|
* the editor screen.
|
|
*/
|
|
public static Intent buildEditorIntent(@NonNull Context context,
|
|
@NonNull List<Media> media,
|
|
@NonNull Recipient recipient,
|
|
@NonNull String body,
|
|
@NonNull TransportOption transport)
|
|
{
|
|
Intent intent = buildGalleryIntent(context, recipient, body, transport);
|
|
intent.putParcelableArrayListExtra(KEY_MEDIA, new ArrayList<>(media));
|
|
return intent;
|
|
}
|
|
|
|
@Override
|
|
protected void onPreCreate() {
|
|
dynamicLanguage.onCreate(this);
|
|
}
|
|
|
|
@Override
|
|
protected void onCreate(Bundle savedInstanceState, boolean ready) {
|
|
setContentView(R.layout.mediasend_activity);
|
|
setResult(RESULT_CANCELED);
|
|
|
|
if (savedInstanceState != null) {
|
|
return;
|
|
}
|
|
|
|
countButton = findViewById(R.id.mediasend_count_button);
|
|
countButtonText = findViewById(R.id.mediasend_count_button_text);
|
|
cameraButton = findViewById(R.id.mediasend_camera_button);
|
|
|
|
viewModel = ViewModelProviders.of(this, new MediaSendViewModel.Factory(getApplication(), new MediaRepository())).get(MediaSendViewModel.class);
|
|
recipient = Recipient.from(this, Address.fromSerialized(getIntent().getStringExtra(KEY_ADDRESS)), true);
|
|
transport = getIntent().getParcelableExtra(KEY_TRANSPORT);
|
|
|
|
viewModel.setTransport(transport);
|
|
viewModel.onBodyChanged(getIntent().getStringExtra(KEY_BODY));
|
|
|
|
List<Media> media = getIntent().getParcelableArrayListExtra(KEY_MEDIA);
|
|
boolean isCamera = getIntent().getBooleanExtra(KEY_IS_CAMERA, false);
|
|
|
|
if (isCamera) {
|
|
Fragment fragment = Camera1Fragment.newInstance();
|
|
getSupportFragmentManager().beginTransaction()
|
|
.replace(R.id.mediasend_fragment_container, fragment, TAG_CAMERA)
|
|
.commit();
|
|
|
|
} else if (!Util.isEmpty(media)) {
|
|
viewModel.onSelectedMediaChanged(this, media);
|
|
|
|
Fragment fragment = MediaSendFragment.newInstance(recipient, transport, dynamicLanguage.getCurrentLocale());
|
|
getSupportFragmentManager().beginTransaction()
|
|
.replace(R.id.mediasend_fragment_container, fragment, TAG_SEND)
|
|
.commit();
|
|
} else {
|
|
MediaPickerFolderFragment fragment = MediaPickerFolderFragment.newInstance(recipient);
|
|
getSupportFragmentManager().beginTransaction()
|
|
.replace(R.id.mediasend_fragment_container, fragment, TAG_FOLDER_PICKER)
|
|
.commit();
|
|
}
|
|
|
|
initializeCountButtonObserver(transport, dynamicLanguage.getCurrentLocale());
|
|
initializeCameraButtonObserver();
|
|
initializeErrorObserver();
|
|
|
|
cameraButton.setOnClickListener(v -> {
|
|
int maxSelection = viewModel.getMaxSelection();
|
|
|
|
if (viewModel.getSelectedMedia().getValue() != null && viewModel.getSelectedMedia().getValue().size() >= maxSelection) {
|
|
Toast.makeText(this, getResources().getQuantityString(R.plurals.MediaSendActivity_cant_share_more_than_n_items, maxSelection, maxSelection), Toast.LENGTH_SHORT).show();
|
|
} else {
|
|
navigateToCamera();
|
|
}
|
|
});
|
|
}
|
|
|
|
@Override
|
|
protected void onResume() {
|
|
super.onResume();
|
|
dynamicLanguage.onResume(this);
|
|
}
|
|
|
|
@Override
|
|
public void onBackPressed() {
|
|
MediaSendFragment sendFragment = (MediaSendFragment) getSupportFragmentManager().findFragmentByTag(TAG_SEND);
|
|
if (sendFragment == null || !sendFragment.isVisible() || !sendFragment.handleBackPress()) {
|
|
super.onBackPressed();
|
|
|
|
if (getIntent().getBooleanExtra(KEY_IS_CAMERA, false) && getSupportFragmentManager().getBackStackEntryCount() == 0) {
|
|
viewModel.onImageCaptureUndo(this);
|
|
}
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
|
|
Permissions.onRequestPermissionsResult(this, requestCode, permissions, grantResults);
|
|
}
|
|
|
|
@Override
|
|
public void onFolderSelected(@NonNull MediaFolder folder) {
|
|
viewModel.onFolderSelected(folder.getBucketId());
|
|
|
|
MediaPickerItemFragment fragment = MediaPickerItemFragment.newInstance(folder.getBucketId(), folder.getTitle(), viewModel.getMaxSelection());
|
|
getSupportFragmentManager().beginTransaction()
|
|
.setCustomAnimations(R.anim.slide_from_right, R.anim.slide_to_left, R.anim.slide_from_left, R.anim.slide_to_right)
|
|
.replace(R.id.mediasend_fragment_container, fragment, TAG_ITEM_PICKER)
|
|
.addToBackStack(null)
|
|
.commit();
|
|
}
|
|
|
|
@Override
|
|
public void onMediaSelected(@NonNull Media media) {
|
|
viewModel.onSingleMediaSelected(this, media);
|
|
navigateToMediaSend(recipient, transport, dynamicLanguage.getCurrentLocale());
|
|
}
|
|
|
|
@Override
|
|
public void onAddMediaClicked(@NonNull String bucketId) {
|
|
// TODO: Get actual folder title somehow
|
|
MediaPickerFolderFragment folderFragment = MediaPickerFolderFragment.newInstance(recipient);
|
|
MediaPickerItemFragment itemFragment = MediaPickerItemFragment.newInstance(bucketId, "", viewModel.getMaxSelection());
|
|
|
|
getSupportFragmentManager().beginTransaction()
|
|
.setCustomAnimations(R.anim.stationary, R.anim.slide_to_left, R.anim.slide_from_left, R.anim.slide_to_right)
|
|
.replace(R.id.mediasend_fragment_container, folderFragment, TAG_FOLDER_PICKER)
|
|
.addToBackStack(null)
|
|
.commit();
|
|
|
|
getSupportFragmentManager().beginTransaction()
|
|
.setCustomAnimations(R.anim.slide_from_right, R.anim.stationary, R.anim.slide_from_left, R.anim.slide_to_right)
|
|
.replace(R.id.mediasend_fragment_container, itemFragment, TAG_ITEM_PICKER)
|
|
.addToBackStack(null)
|
|
.commit();
|
|
}
|
|
|
|
@Override
|
|
public void onSendClicked(@NonNull List<Media> media, @NonNull String message, @NonNull TransportOption transport) {
|
|
viewModel.onSendClicked();
|
|
|
|
ArrayList<Media> mediaList = new ArrayList<>(media);
|
|
Intent intent = new Intent();
|
|
|
|
intent.putParcelableArrayListExtra(EXTRA_MEDIA, mediaList);
|
|
intent.putExtra(EXTRA_MESSAGE, message);
|
|
intent.putExtra(EXTRA_TRANSPORT, transport);
|
|
setResult(RESULT_OK, intent);
|
|
finish();
|
|
|
|
overridePendingTransition(R.anim.stationary, R.anim.camera_slide_to_bottom);
|
|
}
|
|
|
|
@Override
|
|
public void onNoMediaAvailable() {
|
|
setResult(RESULT_CANCELED);
|
|
finish();
|
|
}
|
|
|
|
@Override
|
|
public void onImageEditComplete(@NonNull Uri uri, int width, int height, long size, @NonNull Optional<String> message, @NonNull Optional<TransportOption> transport) {
|
|
throw new UnsupportedOperationException("Callback unsupported.");
|
|
}
|
|
|
|
@Override
|
|
public void onImageEditFailure() {
|
|
throw new UnsupportedOperationException("Callback unsupported.");
|
|
}
|
|
|
|
@Override
|
|
public void onTouchEventsNeeded(boolean needed) {
|
|
MediaSendFragment fragment = (MediaSendFragment) getSupportFragmentManager().findFragmentByTag(TAG_SEND);
|
|
if (fragment != null) {
|
|
fragment.onTouchEventsNeeded(needed);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onCameraError() {
|
|
Toast.makeText(this, R.string.MediaSendActivity_camera_unavailable, Toast.LENGTH_SHORT).show();
|
|
setResult(RESULT_CANCELED, new Intent());
|
|
finish();
|
|
}
|
|
|
|
@Override
|
|
public void onImageCaptured(@NonNull byte[] data, int width, int height) {
|
|
Log.i(TAG, "Camera image captured.");
|
|
|
|
SimpleTask.run(getLifecycle(), () -> {
|
|
try {
|
|
Uri uri = BlobProvider.getInstance()
|
|
.forData(data)
|
|
.withMimeType(MediaUtil.IMAGE_JPEG)
|
|
.createForSingleSessionOnDisk(this, e -> Log.w(TAG, "Failed to write to disk.", e));
|
|
return new Media(uri,
|
|
MediaUtil.IMAGE_JPEG,
|
|
System.currentTimeMillis(),
|
|
width,
|
|
height,
|
|
data.length,
|
|
Optional.of(Media.ALL_MEDIA_BUCKET_ID),
|
|
Optional.absent());
|
|
} catch (IOException e) {
|
|
return null;
|
|
}
|
|
}, media -> {
|
|
if (media == null) {
|
|
onImageEditFailure();
|
|
return;
|
|
}
|
|
|
|
Log.i(TAG, "Camera capture stored: " + media.getUri().toString());
|
|
|
|
viewModel.onImageCaptured(media);
|
|
navigateToMediaSend(recipient, transport, dynamicLanguage.getCurrentLocale());
|
|
});
|
|
}
|
|
|
|
@Override
|
|
public int getDisplayRotation() {
|
|
return getWindowManager().getDefaultDisplay().getRotation();
|
|
}
|
|
|
|
private void initializeCountButtonObserver(@NonNull TransportOption transport, @NonNull Locale locale) {
|
|
viewModel.getCountButtonState().observe(this, buttonState -> {
|
|
if (buttonState == null) return;
|
|
|
|
countButtonText.setText(String.valueOf(buttonState.getCount()));
|
|
countButton.setEnabled(buttonState.isVisible());
|
|
animateButtonVisibility(countButton, countButton.getVisibility(), buttonState.isVisible() ? View.VISIBLE : View.GONE);
|
|
|
|
if (buttonState.getCount() > 0) {
|
|
countButton.setOnClickListener(v -> navigateToMediaSend(recipient, transport, locale));
|
|
if (buttonState.isVisible()) {
|
|
animateButtonTextChange(countButton);
|
|
}
|
|
} else {
|
|
countButton.setOnClickListener(null);
|
|
}
|
|
});
|
|
}
|
|
|
|
private void initializeCameraButtonObserver() {
|
|
viewModel.getCameraButtonVisibility().observe(this, visible -> {
|
|
if (visible == null) return;
|
|
animateButtonVisibility(cameraButton, cameraButton.getVisibility(), visible ? View.VISIBLE : View.GONE);
|
|
});
|
|
}
|
|
|
|
private void initializeErrorObserver() {
|
|
viewModel.getError().observe(this, error -> {
|
|
if (error == null) return;
|
|
|
|
switch (error) {
|
|
case ITEM_TOO_LARGE:
|
|
Toast.makeText(this, R.string.MediaSendActivity_an_item_was_removed_because_it_exceeded_the_size_limit, Toast.LENGTH_LONG).show();
|
|
break;
|
|
case TOO_MANY_ITEMS:
|
|
int maxSelection = viewModel.getMaxSelection();
|
|
Toast.makeText(this, getResources().getQuantityString(R.plurals.MediaSendActivity_cant_share_more_than_n_items, maxSelection, maxSelection), Toast.LENGTH_SHORT).show();
|
|
break;
|
|
}
|
|
});
|
|
}
|
|
|
|
private void navigateToMediaSend(@NonNull Recipient recipient, @NonNull TransportOption transport, @NonNull Locale locale) {
|
|
MediaSendFragment fragment = MediaSendFragment.newInstance(recipient, transport, locale);
|
|
String backstackTag = null;
|
|
|
|
if (getSupportFragmentManager().findFragmentByTag(TAG_SEND) != null) {
|
|
getSupportFragmentManager().popBackStack(TAG_SEND, FragmentManager.POP_BACK_STACK_INCLUSIVE);
|
|
backstackTag = TAG_SEND;
|
|
}
|
|
|
|
getSupportFragmentManager().beginTransaction()
|
|
.setCustomAnimations(R.anim.slide_from_right, R.anim.slide_to_left, R.anim.slide_from_left, R.anim.slide_to_right)
|
|
.replace(R.id.mediasend_fragment_container, fragment, TAG_SEND)
|
|
.addToBackStack(backstackTag)
|
|
.commit();
|
|
}
|
|
|
|
private void navigateToCamera() {
|
|
Permissions.with(this)
|
|
.request(Manifest.permission.CAMERA)
|
|
.ifNecessary()
|
|
.withRationaleDialog(getString(R.string.ConversationActivity_to_capture_photos_and_video_allow_signal_access_to_the_camera), R.drawable.ic_photo_camera_white_48dp)
|
|
.withPermanentDenialDialog(getString(R.string.ConversationActivity_signal_needs_the_camera_permission_to_take_photos_or_video))
|
|
.onAllGranted(() -> {
|
|
Camera1Fragment fragment = getOrCreateCameraFragment();
|
|
getSupportFragmentManager().beginTransaction()
|
|
.setCustomAnimations(R.anim.slide_from_right, R.anim.slide_to_left, R.anim.slide_from_left, R.anim.slide_to_right)
|
|
.replace(R.id.mediasend_fragment_container, fragment, TAG_CAMERA)
|
|
.addToBackStack(null)
|
|
.commit();
|
|
})
|
|
.onAnyDenied(() -> Toast.makeText(MediaSendActivity.this, R.string.ConversationActivity_signal_needs_camera_permissions_to_take_photos_or_video, Toast.LENGTH_LONG).show())
|
|
.execute();
|
|
}
|
|
|
|
private Camera1Fragment getOrCreateCameraFragment() {
|
|
Camera1Fragment fragment = (Camera1Fragment) getSupportFragmentManager().findFragmentByTag(TAG_CAMERA);
|
|
|
|
return fragment != null ? fragment
|
|
: Camera1Fragment.newInstance();
|
|
}
|
|
|
|
private void animateButtonVisibility(@NonNull View button, int oldVisibility, int newVisibility) {
|
|
if (oldVisibility == newVisibility) return;
|
|
|
|
if (button.getAnimation() != null) {
|
|
button.getAnimation().cancel();
|
|
button.setVisibility(newVisibility);
|
|
} else if (newVisibility == View.VISIBLE) {
|
|
button.setVisibility(View.VISIBLE);
|
|
|
|
Animation animation = new ScaleAnimation(0, 1, 0, 1, Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 0.5f);
|
|
animation.setDuration(250);
|
|
animation.setInterpolator(new OvershootInterpolator());
|
|
button.startAnimation(animation);
|
|
} else {
|
|
Animation animation = new ScaleAnimation(1, 0, 1, 0, Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 0.5f);
|
|
animation.setDuration(150);
|
|
animation.setInterpolator(new AccelerateDecelerateInterpolator());
|
|
animation.setAnimationListener(new SimpleAnimationListener() {
|
|
@Override
|
|
public void onAnimationEnd(Animation animation) {
|
|
button.setVisibility(View.GONE);
|
|
}
|
|
});
|
|
|
|
button.startAnimation(animation);
|
|
}
|
|
}
|
|
|
|
private void animateButtonTextChange(@NonNull View button) {
|
|
if (button.getAnimation() != null) {
|
|
button.getAnimation().cancel();
|
|
}
|
|
|
|
Animation grow = new ScaleAnimation(1f, 1.3f, 1f, 1.3f, Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 0.5f);
|
|
grow.setDuration(125);
|
|
grow.setInterpolator(new AccelerateInterpolator());
|
|
grow.setAnimationListener(new SimpleAnimationListener() {
|
|
@Override
|
|
public void onAnimationEnd(Animation animation) {
|
|
Animation shrink = new ScaleAnimation(1.3f, 1f, 1.3f, 1f, Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 0.5f);
|
|
shrink.setDuration(125);
|
|
shrink.setInterpolator(new DecelerateInterpolator());
|
|
button.startAnimation(shrink);
|
|
}
|
|
});
|
|
|
|
button.startAnimation(grow);
|
|
}
|
|
}
|