WIP: preview media before sending them
Note that this commit breaks a few things in that non video/image do not work and sharing media isn’t currently using the new mechanism either
This commit is contained in:
parent
7bde04877a
commit
3608b0eb44
|
@ -54,6 +54,7 @@ import eu.siacs.conversations.entities.DownloadableFile;
|
||||||
import eu.siacs.conversations.entities.Message;
|
import eu.siacs.conversations.entities.Message;
|
||||||
import eu.siacs.conversations.services.XmppConnectionService;
|
import eu.siacs.conversations.services.XmppConnectionService;
|
||||||
import eu.siacs.conversations.ui.RecordingActivity;
|
import eu.siacs.conversations.ui.RecordingActivity;
|
||||||
|
import eu.siacs.conversations.ui.util.Attachment;
|
||||||
import eu.siacs.conversations.utils.CryptoHelper;
|
import eu.siacs.conversations.utils.CryptoHelper;
|
||||||
import eu.siacs.conversations.utils.ExifHelper;
|
import eu.siacs.conversations.utils.ExifHelper;
|
||||||
import eu.siacs.conversations.utils.FileUtils;
|
import eu.siacs.conversations.utils.FileUtils;
|
||||||
|
@ -105,16 +106,19 @@ public class FileBackend {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static boolean allFilesUnderSize(Context context, List<Uri> uris, long max) {
|
public static boolean allFilesUnderSize(Context context, List<Attachment> attachments, long max) {
|
||||||
if (max <= 0) {
|
if (max <= 0) {
|
||||||
Log.d(Config.LOGTAG, "server did not report max file size for http upload");
|
Log.d(Config.LOGTAG, "server did not report max file size for http upload");
|
||||||
return true; //exception to be compatible with HTTP Upload < v0.2
|
return true; //exception to be compatible with HTTP Upload < v0.2
|
||||||
}
|
}
|
||||||
for (Uri uri : uris) {
|
for (Attachment attachment : attachments) {
|
||||||
String mime = context.getContentResolver().getType(uri);
|
if (attachment.getType() != Attachment.Type.FILE) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
String mime = attachment.getMime();
|
||||||
if (mime != null && mime.startsWith("video/")) {
|
if (mime != null && mime.startsWith("video/")) {
|
||||||
try {
|
try {
|
||||||
Dimensions dimensions = FileBackend.getVideoDimensions(context, uri);
|
Dimensions dimensions = FileBackend.getVideoDimensions(context, attachment.getUri());
|
||||||
if (dimensions.getMin() > 720) {
|
if (dimensions.getMin() > 720) {
|
||||||
Log.d(Config.LOGTAG, "do not consider video file with min width larger than 720 for size check");
|
Log.d(Config.LOGTAG, "do not consider video file with min width larger than 720 for size check");
|
||||||
continue;
|
continue;
|
||||||
|
@ -123,7 +127,7 @@ public class FileBackend {
|
||||||
//ignore and fall through
|
//ignore and fall through
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (FileBackend.getFileSize(context, uri) > max) {
|
if (FileBackend.getFileSize(context, attachment.getUri()) > max) {
|
||||||
Log.d(Config.LOGTAG, "not all files are under " + max + " bytes. suggesting falling back to jingle");
|
Log.d(Config.LOGTAG, "not all files are under " + max + " bytes. suggesting falling back to jingle");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -217,6 +221,7 @@ public class FileBackend {
|
||||||
return calcSampleSize(options, size);
|
return calcSampleSize(options, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private static int calcSampleSize(BitmapFactory.Options options, int size) {
|
private static int calcSampleSize(BitmapFactory.Options options, int size) {
|
||||||
int height = options.outHeight;
|
int height = options.outHeight;
|
||||||
int width = options.outWidth;
|
int width = options.outWidth;
|
||||||
|
@ -234,6 +239,29 @@ public class FileBackend {
|
||||||
return inSampleSize;
|
return inSampleSize;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Bitmap getPreviewForUri(Attachment attachment, int size, boolean cacheOnly) {
|
||||||
|
final LruCache<String, Bitmap> cache = mXmppConnectionService.getBitmapCache();
|
||||||
|
Bitmap bitmap = cache.get(attachment.getUuid().toString());
|
||||||
|
if (bitmap != null || cacheOnly) {
|
||||||
|
return bitmap;
|
||||||
|
}
|
||||||
|
Log.d(Config.LOGTAG,"attachment mime="+attachment.getMime());
|
||||||
|
if (attachment.getMime() != null && attachment.getMime().startsWith("video/")) {
|
||||||
|
bitmap = cropCenterSquareVideo(attachment.getUri(), size);
|
||||||
|
drawOverlay(bitmap, paintOverlayBlack(bitmap) ? R.drawable.play_video_black : R.drawable.play_video_white, 0.75f);
|
||||||
|
} else {
|
||||||
|
bitmap = cropCenterSquare(attachment.getUri(), size);
|
||||||
|
if ("image/gif".equals(attachment.getMime())) {
|
||||||
|
Bitmap withGifOverlay = bitmap.copy(Bitmap.Config.ARGB_8888, true);
|
||||||
|
drawOverlay(withGifOverlay, paintOverlayBlack(withGifOverlay) ? R.drawable.play_gif_black : R.drawable.play_gif_white, 1.0f);
|
||||||
|
bitmap.recycle();
|
||||||
|
bitmap = withGifOverlay;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cache.put(attachment.getUuid().toString(), bitmap);
|
||||||
|
return bitmap;
|
||||||
|
}
|
||||||
|
|
||||||
private static Dimensions getVideoDimensions(Context context, Uri uri) throws NotAVideoFile {
|
private static Dimensions getVideoDimensions(Context context, Uri uri) throws NotAVideoFile {
|
||||||
MediaMetadataRetriever mediaMetadataRetriever = new MediaMetadataRetriever();
|
MediaMetadataRetriever mediaMetadataRetriever = new MediaMetadataRetriever();
|
||||||
try {
|
try {
|
||||||
|
@ -733,6 +761,21 @@ public class FileBackend {
|
||||||
return record < 0;
|
return record < 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private Bitmap cropCenterSquareVideo(Uri uri, int size) {
|
||||||
|
MediaMetadataRetriever metadataRetriever = new MediaMetadataRetriever();
|
||||||
|
Bitmap frame;
|
||||||
|
try {
|
||||||
|
metadataRetriever.setDataSource(mXmppConnectionService, uri);
|
||||||
|
frame = metadataRetriever.getFrameAtTime(0);
|
||||||
|
metadataRetriever.release();
|
||||||
|
return cropCenterSquare(frame, size);
|
||||||
|
} catch (Exception e) {
|
||||||
|
frame = Bitmap.createBitmap(size, size, Bitmap.Config.ARGB_8888);
|
||||||
|
frame.eraseColor(0xff000000);
|
||||||
|
return frame;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private Bitmap getVideoPreview(File file, int size) {
|
private Bitmap getVideoPreview(File file, int size) {
|
||||||
MediaMetadataRetriever metadataRetriever = new MediaMetadataRetriever();
|
MediaMetadataRetriever metadataRetriever = new MediaMetadataRetriever();
|
||||||
Bitmap frame;
|
Bitmap frame;
|
||||||
|
|
|
@ -83,9 +83,10 @@ import eu.siacs.conversations.http.HttpDownloadConnection;
|
||||||
import eu.siacs.conversations.persistance.FileBackend;
|
import eu.siacs.conversations.persistance.FileBackend;
|
||||||
import eu.siacs.conversations.services.MessageArchiveService;
|
import eu.siacs.conversations.services.MessageArchiveService;
|
||||||
import eu.siacs.conversations.services.XmppConnectionService;
|
import eu.siacs.conversations.services.XmppConnectionService;
|
||||||
|
import eu.siacs.conversations.ui.adapter.MediaPreviewAdapter;
|
||||||
import eu.siacs.conversations.ui.adapter.MessageAdapter;
|
import eu.siacs.conversations.ui.adapter.MessageAdapter;
|
||||||
import eu.siacs.conversations.ui.util.ActivityResult;
|
import eu.siacs.conversations.ui.util.ActivityResult;
|
||||||
import eu.siacs.conversations.ui.util.AttachmentTool;
|
import eu.siacs.conversations.ui.util.Attachment;
|
||||||
import eu.siacs.conversations.ui.util.ConversationMenuConfigurator;
|
import eu.siacs.conversations.ui.util.ConversationMenuConfigurator;
|
||||||
import eu.siacs.conversations.ui.util.DateSeparator;
|
import eu.siacs.conversations.ui.util.DateSeparator;
|
||||||
import eu.siacs.conversations.ui.util.EditMessageActionModeCallback;
|
import eu.siacs.conversations.ui.util.EditMessageActionModeCallback;
|
||||||
|
@ -151,6 +152,7 @@ public class ConversationFragment extends XmppFragment implements EditMessage.Ke
|
||||||
private final PendingItem<Message> pendingMessage = new PendingItem<>();
|
private final PendingItem<Message> pendingMessage = new PendingItem<>();
|
||||||
public Uri mPendingEditorContent = null;
|
public Uri mPendingEditorContent = null;
|
||||||
protected MessageAdapter messageListAdapter;
|
protected MessageAdapter messageListAdapter;
|
||||||
|
private MediaPreviewAdapter mediaPreviewAdapter;
|
||||||
private String lastMessageUuid = null;
|
private String lastMessageUuid = null;
|
||||||
private Conversation conversation;
|
private Conversation conversation;
|
||||||
private FragmentConversationBinding binding;
|
private FragmentConversationBinding binding;
|
||||||
|
@ -674,7 +676,8 @@ public class ConversationFragment extends XmppFragment implements EditMessage.Ke
|
||||||
}
|
}
|
||||||
|
|
||||||
public void attachEditorContentToConversation(Uri uri) {
|
public void attachEditorContentToConversation(Uri uri) {
|
||||||
this.attachFileToConversation(conversation, uri, null);
|
mediaPreviewAdapter.addMediaPreviews(Attachment.of(getActivity(), uri, Attachment.Type.FILE));
|
||||||
|
toggleInputMethod();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void attachImageToConversation(Conversation conversation, Uri uri) {
|
private void attachImageToConversation(Conversation conversation, Uri uri) {
|
||||||
|
@ -712,6 +715,10 @@ public class ConversationFragment extends XmppFragment implements EditMessage.Ke
|
||||||
}
|
}
|
||||||
|
|
||||||
private void sendMessage() {
|
private void sendMessage() {
|
||||||
|
if (mediaPreviewAdapter.hasAttachments()) {
|
||||||
|
commitAttachments();
|
||||||
|
return;
|
||||||
|
}
|
||||||
final String body = this.binding.textinput.getText().toString();
|
final String body = this.binding.textinput.getText().toString();
|
||||||
final Conversation conversation = this.conversation;
|
final Conversation conversation = this.conversation;
|
||||||
if (body.length() == 0 || conversation == null) {
|
if (body.length() == 0 || conversation == null) {
|
||||||
|
@ -819,16 +826,15 @@ public class ConversationFragment extends XmppFragment implements EditMessage.Ke
|
||||||
selectPresenceToAttachFile(choice);
|
selectPresenceToAttachFile(choice);
|
||||||
break;
|
break;
|
||||||
case ATTACHMENT_CHOICE_CHOOSE_IMAGE:
|
case ATTACHMENT_CHOICE_CHOOSE_IMAGE:
|
||||||
final List<Uri> imageUris = AttachmentTool.extractUriFromIntent(data);
|
final List<Attachment> imageUris = Attachment.extractAttachments(getActivity(), data, Attachment.Type.IMAGE);
|
||||||
for (Iterator<Uri> i = imageUris.iterator(); i.hasNext(); i.remove()) {
|
mediaPreviewAdapter.addMediaPreviews(imageUris);
|
||||||
Log.d(Config.LOGTAG, "ConversationsActivity.onActivityResult() - attaching image to conversations. CHOOSE_IMAGE");
|
toggleInputMethod();
|
||||||
attachImageToConversation(conversation, i.next());
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case ATTACHMENT_CHOICE_TAKE_PHOTO:
|
case ATTACHMENT_CHOICE_TAKE_PHOTO:
|
||||||
final Uri takePhotoUri = pendingTakePhotoUri.pop();
|
final Uri takePhotoUri = pendingTakePhotoUri.pop();
|
||||||
if (takePhotoUri != null) {
|
if (takePhotoUri != null) {
|
||||||
attachImageToConversation(conversation, takePhotoUri);
|
mediaPreviewAdapter.addMediaPreviews(Attachment.of(getActivity(), takePhotoUri, Attachment.Type.IMAGE));
|
||||||
|
toggleInputMethod();
|
||||||
} else {
|
} else {
|
||||||
Log.d(Config.LOGTAG, "lost take photo uri. unable to to attach");
|
Log.d(Config.LOGTAG, "lost take photo uri. unable to to attach");
|
||||||
}
|
}
|
||||||
|
@ -836,19 +842,9 @@ public class ConversationFragment extends XmppFragment implements EditMessage.Ke
|
||||||
case ATTACHMENT_CHOICE_CHOOSE_FILE:
|
case ATTACHMENT_CHOICE_CHOOSE_FILE:
|
||||||
case ATTACHMENT_CHOICE_RECORD_VIDEO:
|
case ATTACHMENT_CHOICE_RECORD_VIDEO:
|
||||||
case ATTACHMENT_CHOICE_RECORD_VOICE:
|
case ATTACHMENT_CHOICE_RECORD_VOICE:
|
||||||
final List<Uri> fileUris = AttachmentTool.extractUriFromIntent(data);
|
final List<Attachment> fileUris = Attachment.extractAttachments(getActivity(), data, Attachment.Type.FILE);
|
||||||
final String type = data == null ? null : data.getType();
|
mediaPreviewAdapter.addMediaPreviews(fileUris);
|
||||||
final PresenceSelector.OnPresenceSelected callback = () -> {
|
toggleInputMethod();
|
||||||
for (Iterator<Uri> i = fileUris.iterator(); i.hasNext(); i.remove()) {
|
|
||||||
Log.d(Config.LOGTAG, "ConversationsActivity.onActivityResult() - attaching file to conversations. CHOOSE_FILE/RECORD_VOICE/RECORD_VIDEO");
|
|
||||||
attachFileToConversation(conversation, i.next(), type);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
if (conversation == null || conversation.getMode() == Conversation.MODE_MULTI || FileBackend.allFilesUnderSize(getActivity(), fileUris, getMaxHttpUploadSize(conversation))) {
|
|
||||||
callback.onPresenceSelected();
|
|
||||||
} else {
|
|
||||||
activity.selectPresence(conversation, callback);
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case ATTACHMENT_CHOICE_LOCATION:
|
case ATTACHMENT_CHOICE_LOCATION:
|
||||||
double latitude = data.getDoubleExtra("latitude", 0);
|
double latitude = data.getDoubleExtra("latitude", 0);
|
||||||
|
@ -868,6 +864,36 @@ public class ConversationFragment extends XmppFragment implements EditMessage.Ke
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void commitAttachments() {
|
||||||
|
final List<Attachment> attachments = mediaPreviewAdapter.getAttachments();
|
||||||
|
final PresenceSelector.OnPresenceSelected callback = () -> {
|
||||||
|
for (Iterator<Attachment> i = attachments.iterator(); i.hasNext(); i.remove()) {
|
||||||
|
final Attachment attachment = i.next();
|
||||||
|
if (attachment.getType() == Attachment.Type.IMAGE) {
|
||||||
|
Log.d(Config.LOGTAG, "ConversationsActivity.commitAttachments() - attaching image to conversations. CHOOSE_IMAGE");
|
||||||
|
attachImageToConversation(conversation, attachment.getUri());
|
||||||
|
} else {
|
||||||
|
Log.d(Config.LOGTAG, "ConversationsActivity.commitAttachments() - attaching file to conversations. CHOOSE_FILE/RECORD_VOICE/RECORD_VIDEO");
|
||||||
|
attachFileToConversation(conversation, attachment.getUri(), attachment.getMime());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
mediaPreviewAdapter.notifyDataSetChanged();
|
||||||
|
toggleInputMethod();
|
||||||
|
};
|
||||||
|
if (conversation == null || conversation.getMode() == Conversation.MODE_MULTI || FileBackend.allFilesUnderSize(getActivity(), attachments, getMaxHttpUploadSize(conversation))) {
|
||||||
|
callback.onPresenceSelected();
|
||||||
|
} else {
|
||||||
|
activity.selectPresence(conversation, callback);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void toggleInputMethod() {
|
||||||
|
boolean hasAttachments = mediaPreviewAdapter.hasAttachments();
|
||||||
|
binding.textinput.setVisibility(hasAttachments ? View.GONE : View.VISIBLE);
|
||||||
|
binding.mediaPreview.setVisibility(hasAttachments ? View.VISIBLE : View.GONE);
|
||||||
|
updateSendButton();
|
||||||
|
}
|
||||||
|
|
||||||
private void handleNegativeActivityResult(int requestCode) {
|
private void handleNegativeActivityResult(int requestCode) {
|
||||||
switch (requestCode) {
|
switch (requestCode) {
|
||||||
//nothing to do for now
|
//nothing to do for now
|
||||||
|
@ -958,6 +984,8 @@ public class ConversationFragment extends XmppFragment implements EditMessage.Ke
|
||||||
binding.scrollToBottomButton.setOnClickListener(this.mScrollButtonListener);
|
binding.scrollToBottomButton.setOnClickListener(this.mScrollButtonListener);
|
||||||
binding.messagesView.setOnScrollListener(mOnScrollListener);
|
binding.messagesView.setOnScrollListener(mOnScrollListener);
|
||||||
binding.messagesView.setTranscriptMode(ListView.TRANSCRIPT_MODE_NORMAL);
|
binding.messagesView.setTranscriptMode(ListView.TRANSCRIPT_MODE_NORMAL);
|
||||||
|
mediaPreviewAdapter = new MediaPreviewAdapter(this);
|
||||||
|
binding.mediaPreview.setAdapter(mediaPreviewAdapter);
|
||||||
messageListAdapter = new MessageAdapter((XmppActivity) getActivity(), this.messageList);
|
messageListAdapter = new MessageAdapter((XmppActivity) getActivity(), this.messageList);
|
||||||
messageListAdapter.setOnContactPictureClicked(message -> {
|
messageListAdapter.setOnContactPictureClicked(message -> {
|
||||||
String fingerprint;
|
String fingerprint;
|
||||||
|
@ -2147,11 +2175,17 @@ public class ConversationFragment extends XmppFragment implements EditMessage.Ke
|
||||||
}
|
}
|
||||||
|
|
||||||
public void updateSendButton() {
|
public void updateSendButton() {
|
||||||
|
boolean hasAttachments = mediaPreviewAdapter != null && mediaPreviewAdapter.hasAttachments();
|
||||||
boolean useSendButtonToIndicateStatus = PreferenceManager.getDefaultSharedPreferences(getActivity()).getBoolean("send_button_status", getResources().getBoolean(R.bool.send_button_status));
|
boolean useSendButtonToIndicateStatus = PreferenceManager.getDefaultSharedPreferences(getActivity()).getBoolean("send_button_status", getResources().getBoolean(R.bool.send_button_status));
|
||||||
final Conversation c = this.conversation;
|
final Conversation c = this.conversation;
|
||||||
final Presence.Status status;
|
final Presence.Status status;
|
||||||
final String text = this.binding.textinput == null ? "" : this.binding.textinput.getText().toString();
|
final String text = this.binding.textinput == null ? "" : this.binding.textinput.getText().toString();
|
||||||
final SendButtonAction action = SendButtonTool.getAction(getActivity(), c, text);
|
final SendButtonAction action;
|
||||||
|
if (hasAttachments) {
|
||||||
|
action = SendButtonAction.TEXT;
|
||||||
|
} else {
|
||||||
|
action = SendButtonTool.getAction(getActivity(), c, text);
|
||||||
|
}
|
||||||
if (useSendButtonToIndicateStatus && c.getAccount().getStatus() == Account.State.ONLINE) {
|
if (useSendButtonToIndicateStatus && c.getAccount().getStatus() == Account.State.ONLINE) {
|
||||||
if (activity.xmppConnectionService != null && activity.xmppConnectionService.getMessageArchiveService().isCatchingUp(c)) {
|
if (activity.xmppConnectionService != null && activity.xmppConnectionService.getMessageArchiveService().isCatchingUp(c)) {
|
||||||
status = Presence.Status.OFFLINE;
|
status = Presence.Status.OFFLINE;
|
||||||
|
|
|
@ -318,7 +318,7 @@ public class ShareWithActivity extends XmppActivity implements XmppConnectionSer
|
||||||
if (account.httpUploadAvailable()
|
if (account.httpUploadAvailable()
|
||||||
&& ((share.image && !neverCompressPictures())
|
&& ((share.image && !neverCompressPictures())
|
||||||
|| conversation.getMode() == Conversation.MODE_MULTI
|
|| conversation.getMode() == Conversation.MODE_MULTI
|
||||||
|| FileBackend.allFilesUnderSize(this, share.uris, max))) {
|
/*|| FileBackend.allFilesUnderSize(this, share.uris, max)*/)) {
|
||||||
callback.onPresenceSelected();
|
callback.onPresenceSelected();
|
||||||
} else {
|
} else {
|
||||||
selectPresence(conversation, callback);
|
selectPresence(conversation, callback);
|
||||||
|
|
|
@ -0,0 +1,179 @@
|
||||||
|
package eu.siacs.conversations.ui.adapter;
|
||||||
|
|
||||||
|
import android.app.Activity;
|
||||||
|
import android.content.res.Resources;
|
||||||
|
import android.databinding.DataBindingUtil;
|
||||||
|
import android.graphics.Bitmap;
|
||||||
|
import android.graphics.drawable.BitmapDrawable;
|
||||||
|
import android.graphics.drawable.Drawable;
|
||||||
|
import android.net.Uri;
|
||||||
|
import android.os.AsyncTask;
|
||||||
|
import android.support.annotation.NonNull;
|
||||||
|
import android.support.v7.widget.RecyclerView;
|
||||||
|
import android.view.LayoutInflater;
|
||||||
|
import android.view.View;
|
||||||
|
import android.view.ViewGroup;
|
||||||
|
import android.widget.ImageView;
|
||||||
|
|
||||||
|
import java.lang.ref.WeakReference;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.RejectedExecutionException;
|
||||||
|
|
||||||
|
import eu.siacs.conversations.R;
|
||||||
|
import eu.siacs.conversations.databinding.MediaPreviewBinding;
|
||||||
|
import eu.siacs.conversations.entities.Conversation;
|
||||||
|
import eu.siacs.conversations.ui.ConversationFragment;
|
||||||
|
import eu.siacs.conversations.ui.XmppActivity;
|
||||||
|
import eu.siacs.conversations.ui.util.Attachment;
|
||||||
|
import eu.siacs.conversations.utils.UIHelper;
|
||||||
|
|
||||||
|
public class MediaPreviewAdapter extends RecyclerView.Adapter<MediaPreviewAdapter.MediaPreviewViewHolder> {
|
||||||
|
|
||||||
|
private final List<Attachment> mediaPreviews = new ArrayList<>();
|
||||||
|
|
||||||
|
private final ConversationFragment conversationFragment;
|
||||||
|
|
||||||
|
public MediaPreviewAdapter(ConversationFragment fragment) {
|
||||||
|
this.conversationFragment = fragment;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NonNull
|
||||||
|
@Override
|
||||||
|
public MediaPreviewViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
|
||||||
|
LayoutInflater layoutInflater = LayoutInflater.from(parent.getContext());
|
||||||
|
MediaPreviewBinding binding = DataBindingUtil.inflate(layoutInflater, R.layout.media_preview, parent, false);
|
||||||
|
return new MediaPreviewViewHolder(binding);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onBindViewHolder(@NonNull MediaPreviewViewHolder holder, int position) {
|
||||||
|
final Attachment attachment = mediaPreviews.get(position);
|
||||||
|
loadPreview(attachment, holder.binding.mediaPreview);
|
||||||
|
holder.binding.deleteButton.setOnClickListener(v -> {
|
||||||
|
int pos = mediaPreviews.indexOf(attachment);
|
||||||
|
mediaPreviews.remove(pos);
|
||||||
|
notifyItemRemoved(pos);
|
||||||
|
conversationFragment.toggleInputMethod();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void addMediaPreviews(List<Attachment> attachments) {
|
||||||
|
this.mediaPreviews.addAll(attachments);
|
||||||
|
notifyDataSetChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void loadPreview(Attachment attachment, ImageView imageView) {
|
||||||
|
if (cancelPotentialWork(attachment, imageView)) {
|
||||||
|
XmppActivity activity = (XmppActivity) conversationFragment.getActivity();
|
||||||
|
final Bitmap bm = activity.xmppConnectionService.getFileBackend().getPreviewForUri(attachment,Math.round(activity.getResources().getDimension(R.dimen.media_preview_size)),true);
|
||||||
|
if (bm != null) {
|
||||||
|
cancelPotentialWork(attachment, imageView);
|
||||||
|
imageView.setImageBitmap(bm);
|
||||||
|
imageView.setBackgroundColor(0x00000000);
|
||||||
|
} else {
|
||||||
|
imageView.setImageDrawable(null);
|
||||||
|
final BitmapWorkerTask task = new BitmapWorkerTask(imageView);
|
||||||
|
final AsyncDrawable asyncDrawable = new AsyncDrawable(conversationFragment.getActivity().getResources(), null, task);
|
||||||
|
imageView.setImageDrawable(asyncDrawable);
|
||||||
|
try {
|
||||||
|
task.execute(attachment);
|
||||||
|
} catch (final RejectedExecutionException ignored) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static boolean cancelPotentialWork(Attachment attachment, ImageView imageView) {
|
||||||
|
final BitmapWorkerTask bitmapWorkerTask = getBitmapWorkerTask(imageView);
|
||||||
|
|
||||||
|
if (bitmapWorkerTask != null) {
|
||||||
|
final Attachment oldAttachment = bitmapWorkerTask.attachment;
|
||||||
|
if (oldAttachment == null || !oldAttachment.equals(attachment)) {
|
||||||
|
bitmapWorkerTask.cancel(true);
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static BitmapWorkerTask getBitmapWorkerTask(ImageView imageView) {
|
||||||
|
if (imageView != null) {
|
||||||
|
final Drawable drawable = imageView.getDrawable();
|
||||||
|
if (drawable instanceof AsyncDrawable) {
|
||||||
|
final AsyncDrawable asyncDrawable = (AsyncDrawable) drawable;
|
||||||
|
return asyncDrawable.getBitmapWorkerTask();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getItemCount() {
|
||||||
|
return mediaPreviews.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean hasAttachments() {
|
||||||
|
return mediaPreviews.size() > 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<Attachment> getAttachments() {
|
||||||
|
return mediaPreviews;
|
||||||
|
}
|
||||||
|
|
||||||
|
class MediaPreviewViewHolder extends RecyclerView.ViewHolder {
|
||||||
|
|
||||||
|
private final MediaPreviewBinding binding;
|
||||||
|
|
||||||
|
MediaPreviewViewHolder(MediaPreviewBinding binding) {
|
||||||
|
super(binding.getRoot());
|
||||||
|
this.binding = binding;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static class AsyncDrawable extends BitmapDrawable {
|
||||||
|
private final WeakReference<BitmapWorkerTask> bitmapWorkerTaskReference;
|
||||||
|
|
||||||
|
AsyncDrawable(Resources res, Bitmap bitmap, BitmapWorkerTask bitmapWorkerTask) {
|
||||||
|
super(res, bitmap);
|
||||||
|
bitmapWorkerTaskReference = new WeakReference<>(bitmapWorkerTask);
|
||||||
|
}
|
||||||
|
|
||||||
|
BitmapWorkerTask getBitmapWorkerTask() {
|
||||||
|
return bitmapWorkerTaskReference.get();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class BitmapWorkerTask extends AsyncTask<Attachment, Void, Bitmap> {
|
||||||
|
private final WeakReference<ImageView> imageViewReference;
|
||||||
|
private Attachment attachment = null;
|
||||||
|
|
||||||
|
BitmapWorkerTask(ImageView imageView) {
|
||||||
|
imageViewReference = new WeakReference<>(imageView);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected Bitmap doInBackground(Attachment... params) {
|
||||||
|
Activity activity = conversationFragment.getActivity();
|
||||||
|
if (activity instanceof XmppActivity) {
|
||||||
|
final XmppActivity xmppActivity = (XmppActivity) activity;
|
||||||
|
this.attachment = params[0];
|
||||||
|
return xmppActivity.xmppConnectionService.getFileBackend().getPreviewForUri(this.attachment, Math.round(xmppActivity.getResources().getDimension(R.dimen.media_preview_size)), false);
|
||||||
|
} else {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onPostExecute(Bitmap bitmap) {
|
||||||
|
if (bitmap != null && !isCancelled()) {
|
||||||
|
final ImageView imageView = imageViewReference.get();
|
||||||
|
if (imageView != null) {
|
||||||
|
imageView.setImageBitmap(bitmap);
|
||||||
|
imageView.setBackgroundColor(0x00000000);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,110 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2018, Daniel Gultsch All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without modification,
|
||||||
|
* are permitted provided that the following conditions are met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright notice, this
|
||||||
|
* list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer in the documentation and/or
|
||||||
|
* other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* 3. Neither the name of the copyright holder nor the names of its contributors
|
||||||
|
* may be used to endorse or promote products derived from this software without
|
||||||
|
* specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
||||||
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
|
||||||
|
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
|
||||||
|
* ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package eu.siacs.conversations.ui.util;
|
||||||
|
|
||||||
|
import android.content.ClipData;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.net.Uri;
|
||||||
|
import android.util.Log;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
import eu.siacs.conversations.Config;
|
||||||
|
import eu.siacs.conversations.utils.MimeUtils;
|
||||||
|
|
||||||
|
public class Attachment {
|
||||||
|
|
||||||
|
public String getMime() {
|
||||||
|
return mime;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Type getType() {
|
||||||
|
return type;
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum Type {
|
||||||
|
FILE, IMAGE
|
||||||
|
}
|
||||||
|
|
||||||
|
private final Uri uri;
|
||||||
|
private final Type type;
|
||||||
|
private final UUID uuid;
|
||||||
|
private final String mime;
|
||||||
|
|
||||||
|
private Attachment(Uri uri, Type type, String mime) {
|
||||||
|
this.uri = uri;
|
||||||
|
this.type = type;
|
||||||
|
this.mime = mime;
|
||||||
|
this.uuid = UUID.randomUUID();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static List<Attachment> of(final Context context, Uri uri, Type type) {
|
||||||
|
final String mime = MimeUtils.guessMimeTypeFromUri(context, uri);
|
||||||
|
return Collections.singletonList(new Attachment(uri, type, mime));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public static List<Attachment> extractAttachments(final Context context, final Intent intent, Type type) {
|
||||||
|
List<Attachment> uris = new ArrayList<>();
|
||||||
|
if (intent == null) {
|
||||||
|
return uris;
|
||||||
|
}
|
||||||
|
final String contentType = intent.getType();
|
||||||
|
final Uri data = intent.getData();
|
||||||
|
if (data == null) {
|
||||||
|
final ClipData clipData = intent.getClipData();
|
||||||
|
if (clipData != null) {
|
||||||
|
for (int i = 0; i < clipData.getItemCount(); ++i) {
|
||||||
|
final Uri uri = clipData.getItemAt(i).getUri();
|
||||||
|
Log.d(Config.LOGTAG,"uri="+uri+" contentType="+contentType);
|
||||||
|
final String mime = contentType != null ? contentType : MimeUtils.guessMimeTypeFromUri(context, uri);
|
||||||
|
Log.d(Config.LOGTAG,"mime="+mime);
|
||||||
|
uris.add(new Attachment(uri, type, mime));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
final String mime = contentType != null ? contentType : MimeUtils.guessMimeTypeFromUri(context, data);
|
||||||
|
uris.add(new Attachment(data, type, mime));
|
||||||
|
}
|
||||||
|
return uris;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Uri getUri() {
|
||||||
|
return uri;
|
||||||
|
}
|
||||||
|
|
||||||
|
public UUID getUuid() {
|
||||||
|
return uuid;
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,61 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) 2018, Daniel Gultsch All rights reserved.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without modification,
|
|
||||||
* are permitted provided that the following conditions are met:
|
|
||||||
*
|
|
||||||
* 1. Redistributions of source code must retain the above copyright notice, this
|
|
||||||
* list of conditions and the following disclaimer.
|
|
||||||
*
|
|
||||||
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
||||||
* this list of conditions and the following disclaimer in the documentation and/or
|
|
||||||
* other materials provided with the distribution.
|
|
||||||
*
|
|
||||||
* 3. Neither the name of the copyright holder nor the names of its contributors
|
|
||||||
* may be used to endorse or promote products derived from this software without
|
|
||||||
* specific prior written permission.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
||||||
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
|
|
||||||
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
|
|
||||||
* ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package eu.siacs.conversations.ui.util;
|
|
||||||
|
|
||||||
import android.annotation.SuppressLint;
|
|
||||||
import android.content.ClipData;
|
|
||||||
import android.content.Intent;
|
|
||||||
import android.net.Uri;
|
|
||||||
import android.os.Build;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
public class AttachmentTool {
|
|
||||||
@SuppressLint("NewApi")
|
|
||||||
public static List<Uri> extractUriFromIntent(final Intent intent) {
|
|
||||||
List<Uri> uris = new ArrayList<>();
|
|
||||||
if (intent == null) {
|
|
||||||
return uris;
|
|
||||||
}
|
|
||||||
final Uri uri = intent.getData();
|
|
||||||
if (uri == null) {
|
|
||||||
final ClipData clipData = intent.getClipData();
|
|
||||||
if (clipData != null) {
|
|
||||||
for (int i = 0; i < clipData.getItemCount(); ++i) {
|
|
||||||
uris.add(clipData.getItemAt(i).getUri());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
uris.add(uri);
|
|
||||||
}
|
|
||||||
return uris;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -13,7 +13,7 @@
|
||||||
android:layout_width="fill_parent"
|
android:layout_width="fill_parent"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:layout_above="@+id/snackbar"
|
android:layout_above="@+id/snackbar"
|
||||||
android:layout_alignParentLeft="true"
|
android:layout_alignParentStart="true"
|
||||||
android:layout_alignParentTop="true"
|
android:layout_alignParentTop="true"
|
||||||
android:background="?attr/color_background_secondary"
|
android:background="?attr/color_background_secondary"
|
||||||
android:divider="@null"
|
android:divider="@null"
|
||||||
|
@ -54,15 +54,30 @@
|
||||||
android:layout_width="fill_parent"
|
android:layout_width="fill_parent"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:layout_alignParentBottom="true"
|
android:layout_alignParentBottom="true"
|
||||||
android:layout_alignParentLeft="true"
|
android:layout_alignParentStart="true"
|
||||||
android:background="?attr/color_background_primary">
|
android:background="?attr/color_background_primary">
|
||||||
|
|
||||||
|
<android.support.v7.widget.RecyclerView
|
||||||
|
android:visibility="gone"
|
||||||
|
android:id="@+id/media_preview"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_alignParentStart="true"
|
||||||
|
android:layout_toStartOf="@+id/textSendButton"
|
||||||
|
tools:listitem="@layout/media_preview"
|
||||||
|
android:orientation="horizontal"
|
||||||
|
app:layoutManager="android.support.v7.widget.LinearLayoutManager"
|
||||||
|
android:paddingTop="8dp"
|
||||||
|
android:requiresFadingEdge="horizontal">
|
||||||
|
|
||||||
|
</android.support.v7.widget.RecyclerView>
|
||||||
|
|
||||||
<eu.siacs.conversations.ui.widget.EditMessage
|
<eu.siacs.conversations.ui.widget.EditMessage
|
||||||
android:id="@+id/textinput"
|
android:id="@+id/textinput"
|
||||||
android:layout_width="wrap_content"
|
android:layout_width="wrap_content"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:layout_alignParentLeft="true"
|
android:layout_alignParentStart="true"
|
||||||
android:layout_toLeftOf="@+id/textSendButton"
|
android:layout_toStartOf="@+id/textSendButton"
|
||||||
android:background="?attr/color_background_primary"
|
android:background="?attr/color_background_primary"
|
||||||
android:ems="10"
|
android:ems="10"
|
||||||
style="@style/Widget.Conversations.EditText"
|
style="@style/Widget.Conversations.EditText"
|
||||||
|
@ -83,7 +98,7 @@
|
||||||
android:id="@+id/textSendButton"
|
android:id="@+id/textSendButton"
|
||||||
android:layout_width="48dp"
|
android:layout_width="48dp"
|
||||||
android:layout_height="48dp"
|
android:layout_height="48dp"
|
||||||
android:layout_alignParentRight="true"
|
android:layout_alignParentEnd="true"
|
||||||
android:layout_centerVertical="true"
|
android:layout_centerVertical="true"
|
||||||
android:background="?attr/color_background_primary"
|
android:background="?attr/color_background_primary"
|
||||||
android:contentDescription="@string/send_message"
|
android:contentDescription="@string/send_message"
|
||||||
|
@ -106,17 +121,17 @@
|
||||||
android:id="@+id/snackbar_message"
|
android:id="@+id/snackbar_message"
|
||||||
android:layout_width="wrap_content"
|
android:layout_width="wrap_content"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:layout_alignParentLeft="true"
|
android:layout_alignParentStart="true"
|
||||||
android:layout_centerVertical="true"
|
android:layout_centerVertical="true"
|
||||||
android:layout_toLeftOf="@+id/snackbar_action"
|
android:layout_toStartOf="@+id/snackbar_action"
|
||||||
android:paddingLeft="24dp"
|
android:paddingStart="24dp"
|
||||||
android:textAppearance="@style/TextAppearance.Conversations.Body1.OnDark"/>
|
android:textAppearance="@style/TextAppearance.Conversations.Body1.OnDark"/>
|
||||||
|
|
||||||
<TextView
|
<TextView
|
||||||
android:id="@+id/snackbar_action"
|
android:id="@+id/snackbar_action"
|
||||||
android:layout_width="wrap_content"
|
android:layout_width="wrap_content"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:layout_alignParentRight="true"
|
android:layout_alignParentEnd="true"
|
||||||
android:layout_centerVertical="true"
|
android:layout_centerVertical="true"
|
||||||
android:paddingBottom="16dp"
|
android:paddingBottom="16dp"
|
||||||
android:paddingLeft="24dp"
|
android:paddingLeft="24dp"
|
||||||
|
|
|
@ -0,0 +1,23 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<layout xmlns:android="http://schemas.android.com/apk/res/android">
|
||||||
|
<RelativeLayout
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content">
|
||||||
|
<ImageView
|
||||||
|
android:id="@+id/media_preview"
|
||||||
|
android:layout_width="@dimen/media_preview_size"
|
||||||
|
android:layout_height="@dimen/media_preview_size"
|
||||||
|
android:background="@color/black54"
|
||||||
|
android:layout_centerInParent="true"
|
||||||
|
android:layout_margin="12dp"/>
|
||||||
|
<ImageButton
|
||||||
|
android:id="@+id/delete_button"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_alignParentEnd="true"
|
||||||
|
android:layout_alignParentTop="true"
|
||||||
|
android:alpha="?attr/delete_icon_alpha"
|
||||||
|
android:background="?attr/selectableItemBackgroundBorderless"
|
||||||
|
android:src="?attr/icon_cancel"/>
|
||||||
|
</RelativeLayout>
|
||||||
|
</layout>
|
|
@ -50,6 +50,7 @@
|
||||||
<attr name="conversations_overview_background" format="reference|color"/>
|
<attr name="conversations_overview_background" format="reference|color"/>
|
||||||
|
|
||||||
<attr name="icon_alpha" format="float"/>
|
<attr name="icon_alpha" format="float"/>
|
||||||
|
<attr name="delete_icon_alpha" format="float"/>
|
||||||
|
|
||||||
<attr name="icon_add_group" format="reference"/>
|
<attr name="icon_add_group" format="reference"/>
|
||||||
<attr name="icon_add_person" format="reference"/>
|
<attr name="icon_add_person" format="reference"/>
|
||||||
|
|
|
@ -11,6 +11,7 @@
|
||||||
<dimen name="audio_player_width">224dp</dimen>
|
<dimen name="audio_player_width">224dp</dimen>
|
||||||
<dimen name="avatar_item_distance">16dp</dimen>
|
<dimen name="avatar_item_distance">16dp</dimen>
|
||||||
|
|
||||||
|
<dimen name="media_preview_size">80dp</dimen>
|
||||||
<dimen name="toolbar_elevation">4dp</dimen>
|
<dimen name="toolbar_elevation">4dp</dimen>
|
||||||
|
|
||||||
<dimen name="publish_avatar_top_margin">8dp</dimen>
|
<dimen name="publish_avatar_top_margin">8dp</dimen>
|
||||||
|
|
|
@ -58,6 +58,7 @@
|
||||||
<item name="conversations_overview_background">@color/green700</item>
|
<item name="conversations_overview_background">@color/green700</item>
|
||||||
|
|
||||||
<item type="float" name="icon_alpha">0.54</item>
|
<item type="float" name="icon_alpha">0.54</item>
|
||||||
|
<item type="float" name="delete_icon_alpha">0.70</item>
|
||||||
|
|
||||||
<item name="dialog_horizontal_padding">24dp</item>
|
<item name="dialog_horizontal_padding">24dp</item>
|
||||||
<item name="dialog_vertical_padding">16dp</item>
|
<item name="dialog_vertical_padding">16dp</item>
|
||||||
|
@ -158,6 +159,7 @@
|
||||||
<item name="conversations_overview_background">@color/green900</item>
|
<item name="conversations_overview_background">@color/green900</item>
|
||||||
|
|
||||||
<item type="float" name="icon_alpha">0.7</item>
|
<item type="float" name="icon_alpha">0.7</item>
|
||||||
|
<item type="float" name="delete_icon_alpha">0.7</item>
|
||||||
|
|
||||||
<item name="dialog_horizontal_padding">24dp</item>
|
<item name="dialog_horizontal_padding">24dp</item>
|
||||||
<item name="dialog_vertical_padding">16dp</item>
|
<item name="dialog_vertical_padding">16dp</item>
|
||||||
|
|
Loading…
Reference in New Issue