mirror of
https://github.com/oxen-io/session-android.git
synced 2025-10-24 14:40:10 +00:00
thumbnail generation and disk caching
// FREEBIE
This commit is contained in:
@@ -12,4 +12,5 @@
|
|||||||
<dimen name="conversation_item_corner_radius">3dp</dimen>
|
<dimen name="conversation_item_corner_radius">3dp</dimen>
|
||||||
<dimen name="conversation_item_drop_shadow_dist">2dp</dimen>
|
<dimen name="conversation_item_drop_shadow_dist">2dp</dimen>
|
||||||
<dimen name="contact_selection_photo_size">50dp</dimen>
|
<dimen name="contact_selection_photo_size">50dp</dimen>
|
||||||
|
<dimen name="thumbnail_max_size">230dp</dimen>
|
||||||
</resources>
|
</resources>
|
||||||
|
@@ -58,8 +58,8 @@ public class DatabaseFactory {
|
|||||||
private static final int INTRODUCED_PUSH_FIX_VERSION = 12;
|
private static final int INTRODUCED_PUSH_FIX_VERSION = 12;
|
||||||
private static final int INTRODUCED_DELIVERY_RECEIPTS = 13;
|
private static final int INTRODUCED_DELIVERY_RECEIPTS = 13;
|
||||||
private static final int INTRODUCED_PART_DATA_SIZE_VERSION = 14;
|
private static final int INTRODUCED_PART_DATA_SIZE_VERSION = 14;
|
||||||
private static final int DATABASE_VERSION = 14;
|
private static final int INTRODUCED_THUMBNAILS_VERSION = 15;
|
||||||
|
private static final int DATABASE_VERSION = 15;
|
||||||
|
|
||||||
private static final String DATABASE_NAME = "messages.db";
|
private static final String DATABASE_NAME = "messages.db";
|
||||||
private static final Object lock = new Object();
|
private static final Object lock = new Object();
|
||||||
@@ -705,6 +705,11 @@ public class DatabaseFactory {
|
|||||||
db.execSQL("ALTER TABLE part ADD COLUMN data_size INTEGER DEFAULT 0;");
|
db.execSQL("ALTER TABLE part ADD COLUMN data_size INTEGER DEFAULT 0;");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (oldVersion < INTRODUCED_THUMBNAILS_VERSION) {
|
||||||
|
db.execSQL("ALTER TABLE part ADD COLUMN thumbnail TEXT");
|
||||||
|
db.execSQL("ALTER TABLE part ADD COLUMN aspect_ratio REAL");
|
||||||
|
}
|
||||||
|
|
||||||
db.setTransactionSuccessful();
|
db.setTransactionSuccessful();
|
||||||
db.endTransaction();
|
db.endTransaction();
|
||||||
}
|
}
|
||||||
|
@@ -26,9 +26,11 @@ import android.text.TextUtils;
|
|||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
import android.util.Pair;
|
import android.util.Pair;
|
||||||
|
|
||||||
|
import org.thoughtcrime.securesms.ApplicationContext;
|
||||||
import org.thoughtcrime.securesms.crypto.DecryptingPartInputStream;
|
import org.thoughtcrime.securesms.crypto.DecryptingPartInputStream;
|
||||||
import org.thoughtcrime.securesms.crypto.EncryptingPartOutputStream;
|
import org.thoughtcrime.securesms.crypto.EncryptingPartOutputStream;
|
||||||
import org.thoughtcrime.securesms.crypto.MasterSecret;
|
import org.thoughtcrime.securesms.crypto.MasterSecret;
|
||||||
|
import org.thoughtcrime.securesms.jobs.ThumbnailGenerateJob;
|
||||||
import org.thoughtcrime.securesms.mms.PartAuthority;
|
import org.thoughtcrime.securesms.mms.PartAuthority;
|
||||||
import org.thoughtcrime.securesms.util.Util;
|
import org.thoughtcrime.securesms.util.Util;
|
||||||
|
|
||||||
@@ -66,6 +68,8 @@ public class PartDatabase extends Database {
|
|||||||
private static final String DATA = "_data";
|
private static final String DATA = "_data";
|
||||||
private static final String PENDING_PUSH_ATTACHMENT = "pending_push";
|
private static final String PENDING_PUSH_ATTACHMENT = "pending_push";
|
||||||
private static final String SIZE = "data_size";
|
private static final String SIZE = "data_size";
|
||||||
|
private static final String THUMBNAIL = "thumbnail";
|
||||||
|
private static final String ASPECT_RATIO = "aspect_ratio";
|
||||||
|
|
||||||
public static final String CREATE_TABLE = "CREATE TABLE " + TABLE_NAME + " (" + ID + " INTEGER PRIMARY KEY, " +
|
public static final String CREATE_TABLE = "CREATE TABLE " + TABLE_NAME + " (" + ID + " INTEGER PRIMARY KEY, " +
|
||||||
MMS_ID + " INTEGER, " + SEQUENCE + " INTEGER DEFAULT 0, " +
|
MMS_ID + " INTEGER, " + SEQUENCE + " INTEGER DEFAULT 0, " +
|
||||||
@@ -73,7 +77,8 @@ public class PartDatabase extends Database {
|
|||||||
CONTENT_DISPOSITION + " TEXT, " + FILENAME + " TEXT, " + CONTENT_ID + " TEXT, " +
|
CONTENT_DISPOSITION + " TEXT, " + FILENAME + " TEXT, " + CONTENT_ID + " TEXT, " +
|
||||||
CONTENT_LOCATION + " TEXT, " + CONTENT_TYPE_START + " INTEGER, " +
|
CONTENT_LOCATION + " TEXT, " + CONTENT_TYPE_START + " INTEGER, " +
|
||||||
CONTENT_TYPE_TYPE + " TEXT, " + ENCRYPTED + " INTEGER, " +
|
CONTENT_TYPE_TYPE + " TEXT, " + ENCRYPTED + " INTEGER, " +
|
||||||
PENDING_PUSH_ATTACHMENT + " INTEGER, "+ DATA + " TEXT, " + SIZE + " INTEGER);";
|
PENDING_PUSH_ATTACHMENT + " INTEGER, "+ DATA + " TEXT, " + SIZE + " INTEGER, " +
|
||||||
|
THUMBNAIL + " TEXT, " + ASPECT_RATIO + " REAL);";
|
||||||
|
|
||||||
public static final String[] CREATE_INDEXS = {
|
public static final String[] CREATE_INDEXS = {
|
||||||
"CREATE INDEX IF NOT EXISTS part_mms_id_index ON " + TABLE_NAME + " (" + MMS_ID + ");",
|
"CREATE INDEX IF NOT EXISTS part_mms_id_index ON " + TABLE_NAME + " (" + MMS_ID + ");",
|
||||||
@@ -90,6 +95,12 @@ public class PartDatabase extends Database {
|
|||||||
return getDataStream(masterSecret, partId, DATA);
|
return getDataStream(masterSecret, partId, DATA);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public InputStream getThumbnailStream(MasterSecret masterSecret, long partId)
|
||||||
|
throws FileNotFoundException
|
||||||
|
{
|
||||||
|
return getDataStream(masterSecret, partId, THUMBNAIL);
|
||||||
|
}
|
||||||
|
|
||||||
public void updateFailedDownloadedPart(long messageId, long partId, PduPart part)
|
public void updateFailedDownloadedPart(long messageId, long partId, PduPart part)
|
||||||
throws MmsException
|
throws MmsException
|
||||||
{
|
{
|
||||||
@@ -134,7 +145,7 @@ public class PartDatabase extends Database {
|
|||||||
while (cursor != null && cursor.moveToNext()) {
|
while (cursor != null && cursor.moveToNext()) {
|
||||||
PduPart part = getPart(cursor);
|
PduPart part = getPart(cursor);
|
||||||
results.add(new Pair<>(cursor.getLong(cursor.getColumnIndexOrThrow(ID)),
|
results.add(new Pair<>(cursor.getLong(cursor.getColumnIndexOrThrow(ID)),
|
||||||
part));
|
part));
|
||||||
}
|
}
|
||||||
|
|
||||||
return results;
|
return results;
|
||||||
@@ -144,20 +155,26 @@ public class PartDatabase extends Database {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("ResultOfMethodCallIgnored")
|
||||||
public void deleteParts(long mmsId) {
|
public void deleteParts(long mmsId) {
|
||||||
SQLiteDatabase database = databaseHelper.getWritableDatabase();
|
SQLiteDatabase database = databaseHelper.getWritableDatabase();
|
||||||
Cursor cursor = null;
|
Cursor cursor = null;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
cursor = database.query(TABLE_NAME, new String[] {DATA}, MMS_ID + " = ?",
|
cursor = database.query(TABLE_NAME, new String[] {DATA, THUMBNAIL}, MMS_ID + " = ?",
|
||||||
new String[] {mmsId+""}, null, null, null);
|
new String[] {mmsId+""}, null, null, null);
|
||||||
|
|
||||||
while (cursor != null && cursor.moveToNext()) {
|
while (cursor != null && cursor.moveToNext()) {
|
||||||
String data = cursor.getString(0);
|
String data = cursor.getString(0);
|
||||||
|
String thumbnail = cursor.getString(1);
|
||||||
|
|
||||||
if (!TextUtils.isEmpty(data)) {
|
if (!TextUtils.isEmpty(data)) {
|
||||||
new File(data).delete();
|
new File(data).delete();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!TextUtils.isEmpty(thumbnail)) {
|
||||||
|
new File(thumbnail).delete();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} finally {
|
} finally {
|
||||||
if (cursor != null)
|
if (cursor != null)
|
||||||
@@ -167,6 +184,7 @@ public class PartDatabase extends Database {
|
|||||||
database.delete(TABLE_NAME, MMS_ID + " = ?", new String[] {mmsId+""});
|
database.delete(TABLE_NAME, MMS_ID + " = ?", new String[] {mmsId+""});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("ResultOfMethodCallIgnored")
|
||||||
public void deleteAllParts() {
|
public void deleteAllParts() {
|
||||||
SQLiteDatabase database = databaseHelper.getWritableDatabase();
|
SQLiteDatabase database = databaseHelper.getWritableDatabase();
|
||||||
database.delete(TABLE_NAME, null, null);
|
database.delete(TABLE_NAME, null, null);
|
||||||
@@ -232,6 +250,12 @@ public class PartDatabase extends Database {
|
|||||||
if (!cursor.isNull(pendingPushColumn))
|
if (!cursor.isNull(pendingPushColumn))
|
||||||
part.setPendingPush(cursor.getInt(pendingPushColumn) == 1);
|
part.setPendingPush(cursor.getInt(pendingPushColumn) == 1);
|
||||||
|
|
||||||
|
int thumbnailColumn = cursor.getColumnIndexOrThrow(THUMBNAIL);
|
||||||
|
|
||||||
|
if (!cursor.isNull(thumbnailColumn))
|
||||||
|
part.setThumbnailUri(ContentUris.withAppendedId(PartAuthority.THUMB_CONTENT_URI,
|
||||||
|
cursor.getLong(cursor.getColumnIndexOrThrow(ID))));
|
||||||
|
|
||||||
int sizeColumn = cursor.getColumnIndexOrThrow(SIZE);
|
int sizeColumn = cursor.getColumnIndexOrThrow(SIZE);
|
||||||
|
|
||||||
if (!cursor.isNull(sizeColumn))
|
if (!cursor.isNull(sizeColumn))
|
||||||
@@ -356,9 +380,8 @@ public class PartDatabase extends Database {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private PduPart getPart(Cursor cursor) {
|
private PduPart getPart(Cursor cursor) {
|
||||||
PduPart part = new PduPart();
|
PduPart part = new PduPart();
|
||||||
String dataLocation = cursor.getString(cursor.getColumnIndexOrThrow(DATA));
|
long partId = cursor.getLong(cursor.getColumnIndexOrThrow(ID));
|
||||||
long partId = cursor.getLong(cursor.getColumnIndexOrThrow(ID));
|
|
||||||
|
|
||||||
getPartValues(part, cursor);
|
getPartValues(part, cursor);
|
||||||
|
|
||||||
@@ -385,7 +408,11 @@ public class PartDatabase extends Database {
|
|||||||
contentValues.put(SIZE, partData.second);
|
contentValues.put(SIZE, partData.second);
|
||||||
}
|
}
|
||||||
|
|
||||||
return database.insert(TABLE_NAME, null, contentValues);
|
long partId = database.insert(TABLE_NAME, null, contentValues);
|
||||||
|
|
||||||
|
ApplicationContext.getInstance(context).getJobManager().add(new ThumbnailGenerateJob(context, partId));
|
||||||
|
|
||||||
|
return partId;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void updateDownloadedPart(MasterSecret masterSecret, long messageId,
|
public void updateDownloadedPart(MasterSecret masterSecret, long messageId,
|
||||||
@@ -407,6 +434,24 @@ public class PartDatabase extends Database {
|
|||||||
|
|
||||||
database.update(TABLE_NAME, values, ID_WHERE, new String[] {partId+""});
|
database.update(TABLE_NAME, values, ID_WHERE, new String[] {partId+""});
|
||||||
|
|
||||||
|
ApplicationContext.getInstance(context).getJobManager().add(new ThumbnailGenerateJob(context, partId));
|
||||||
|
|
||||||
notifyConversationListeners(DatabaseFactory.getMmsDatabase(context).getThreadIdForMessage(messageId));
|
notifyConversationListeners(DatabaseFactory.getMmsDatabase(context).getThreadIdForMessage(messageId));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void updatePartThumbnail(MasterSecret masterSecret, long partId, PduPart part, InputStream in, float aspectRatio)
|
||||||
|
throws MmsException
|
||||||
|
{
|
||||||
|
Log.w(TAG, "updating part thumbnail for #" + partId);
|
||||||
|
|
||||||
|
Pair<File, Long> thumbnailFile = writePartData(masterSecret, part, in);
|
||||||
|
|
||||||
|
SQLiteDatabase database = databaseHelper.getWritableDatabase();
|
||||||
|
ContentValues values = new ContentValues(2);
|
||||||
|
|
||||||
|
values.put(THUMBNAIL, thumbnailFile.first.getAbsolutePath());
|
||||||
|
values.put(ASPECT_RATIO, aspectRatio);
|
||||||
|
|
||||||
|
database.update(TABLE_NAME, values, ID_WHERE, new String[] {partId + ""});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
125
src/org/thoughtcrime/securesms/jobs/ThumbnailGenerateJob.java
Normal file
125
src/org/thoughtcrime/securesms/jobs/ThumbnailGenerateJob.java
Normal file
@@ -0,0 +1,125 @@
|
|||||||
|
package org.thoughtcrime.securesms.jobs;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.graphics.Bitmap;
|
||||||
|
import android.graphics.Bitmap.CompressFormat;
|
||||||
|
import android.util.Log;
|
||||||
|
import android.util.Pair;
|
||||||
|
|
||||||
|
import org.thoughtcrime.securesms.R;
|
||||||
|
import org.thoughtcrime.securesms.crypto.AsymmetricMasterCipher;
|
||||||
|
import org.thoughtcrime.securesms.crypto.AsymmetricMasterSecret;
|
||||||
|
import org.thoughtcrime.securesms.crypto.MasterSecret;
|
||||||
|
import org.thoughtcrime.securesms.crypto.MasterSecretUtil;
|
||||||
|
import org.thoughtcrime.securesms.crypto.SecurityEvent;
|
||||||
|
import org.thoughtcrime.securesms.crypto.SmsCipher;
|
||||||
|
import org.thoughtcrime.securesms.crypto.storage.TextSecureAxolotlStore;
|
||||||
|
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
||||||
|
import org.thoughtcrime.securesms.database.EncryptingSmsDatabase;
|
||||||
|
import org.thoughtcrime.securesms.database.NoSuchMessageException;
|
||||||
|
import org.thoughtcrime.securesms.database.PartDatabase;
|
||||||
|
import org.thoughtcrime.securesms.database.model.SmsMessageRecord;
|
||||||
|
import org.thoughtcrime.securesms.jobs.requirements.MasterSecretRequirement;
|
||||||
|
import org.thoughtcrime.securesms.notifications.MessageNotifier;
|
||||||
|
import org.thoughtcrime.securesms.service.KeyCachingService;
|
||||||
|
import org.thoughtcrime.securesms.sms.IncomingEncryptedMessage;
|
||||||
|
import org.thoughtcrime.securesms.sms.IncomingEndSessionMessage;
|
||||||
|
import org.thoughtcrime.securesms.sms.IncomingKeyExchangeMessage;
|
||||||
|
import org.thoughtcrime.securesms.sms.IncomingPreKeyBundleMessage;
|
||||||
|
import org.thoughtcrime.securesms.sms.IncomingTextMessage;
|
||||||
|
import org.thoughtcrime.securesms.sms.MessageSender;
|
||||||
|
import org.thoughtcrime.securesms.sms.OutgoingKeyExchangeMessage;
|
||||||
|
import org.thoughtcrime.securesms.util.BitmapDecodingException;
|
||||||
|
import org.thoughtcrime.securesms.util.BitmapUtil;
|
||||||
|
import org.thoughtcrime.securesms.util.TextSecurePreferences;
|
||||||
|
import org.whispersystems.jobqueue.JobParameters;
|
||||||
|
import org.whispersystems.libaxolotl.DuplicateMessageException;
|
||||||
|
import org.whispersystems.libaxolotl.InvalidMessageException;
|
||||||
|
import org.whispersystems.libaxolotl.InvalidVersionException;
|
||||||
|
import org.whispersystems.libaxolotl.LegacyMessageException;
|
||||||
|
import org.whispersystems.libaxolotl.NoSessionException;
|
||||||
|
import org.whispersystems.libaxolotl.StaleKeyExchangeException;
|
||||||
|
import org.whispersystems.libaxolotl.UntrustedIdentityException;
|
||||||
|
import org.whispersystems.libaxolotl.util.guava.Optional;
|
||||||
|
import org.whispersystems.textsecure.api.messages.TextSecureGroup;
|
||||||
|
|
||||||
|
import java.io.ByteArrayInputStream;
|
||||||
|
import java.io.ByteArrayOutputStream;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
import ws.com.google.android.mms.ContentType;
|
||||||
|
import ws.com.google.android.mms.MmsException;
|
||||||
|
import ws.com.google.android.mms.pdu.PduPart;
|
||||||
|
|
||||||
|
public class ThumbnailGenerateJob extends MasterSecretJob {
|
||||||
|
|
||||||
|
private static final String TAG = ThumbnailGenerateJob.class.getSimpleName();
|
||||||
|
|
||||||
|
private final long partId;
|
||||||
|
|
||||||
|
public ThumbnailGenerateJob(Context context, long partId) {
|
||||||
|
super(context, JobParameters.newBuilder()
|
||||||
|
.withRequirement(new MasterSecretRequirement(context))
|
||||||
|
.create());
|
||||||
|
|
||||||
|
this.partId = partId;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onAdded() { }
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onRun(MasterSecret masterSecret) throws MmsException {
|
||||||
|
PartDatabase database = DatabaseFactory.getPartDatabase(context);
|
||||||
|
PduPart part = database.getPart(partId);
|
||||||
|
|
||||||
|
if (part.getThumbnailUri() != null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
long startMillis = System.currentTimeMillis();
|
||||||
|
Bitmap thumbnail = generateThumbnailForPart(masterSecret, part);
|
||||||
|
|
||||||
|
if (thumbnail != null) {
|
||||||
|
ByteArrayOutputStream thumbnailBytes = new ByteArrayOutputStream();
|
||||||
|
thumbnail.compress(CompressFormat.JPEG, 85, thumbnailBytes);
|
||||||
|
|
||||||
|
float aspectRatio = (float)thumbnail.getWidth() / (float)thumbnail.getHeight();
|
||||||
|
Log.w(TAG, String.format("generated thumbnail for part #%d, %dx%d (%.3f:1) in %dms",
|
||||||
|
partId,
|
||||||
|
thumbnail.getWidth(),
|
||||||
|
thumbnail.getHeight(),
|
||||||
|
aspectRatio, System.currentTimeMillis() - startMillis));
|
||||||
|
database.updatePartThumbnail(masterSecret, partId, part, new ByteArrayInputStream(thumbnailBytes.toByteArray()), aspectRatio);
|
||||||
|
} else {
|
||||||
|
Log.w(TAG, "thumbnail not generated");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private Bitmap generateThumbnailForPart(MasterSecret masterSecret, PduPart part) {
|
||||||
|
String contentType = new String(part.getContentType());
|
||||||
|
|
||||||
|
if (ContentType.isImageType(contentType)) return generateImageThumbnail(masterSecret, part);
|
||||||
|
else return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Bitmap generateImageThumbnail(MasterSecret masterSecret, PduPart part) {
|
||||||
|
try {
|
||||||
|
int maxSize = context.getResources().getDimensionPixelSize(R.dimen.thumbnail_max_size);
|
||||||
|
return BitmapUtil.createScaledBitmap(context, masterSecret, part.getDataUri(), maxSize, maxSize);
|
||||||
|
} catch (FileNotFoundException | BitmapDecodingException e) {
|
||||||
|
Log.w(TAG, e);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean onShouldRetryThrowable(Exception exception) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onCanceled() { }
|
||||||
|
}
|
@@ -20,7 +20,9 @@ import android.app.Activity;
|
|||||||
import android.content.ActivityNotFoundException;
|
import android.content.ActivityNotFoundException;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
|
import android.graphics.drawable.Drawable;
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
|
import android.os.AsyncTask;
|
||||||
import android.os.Build;
|
import android.os.Build;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
import android.provider.ContactsContract;
|
import android.provider.ContactsContract;
|
||||||
@@ -74,12 +76,23 @@ public class AttachmentManager {
|
|||||||
setMedia(new AudioSlide(context, audio));
|
setMedia(new AudioSlide(context, audio));
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setMedia(Slide slide, int thumbnailWidth, int thumbnailHeight) {
|
public void setMedia(final Slide slide, final int thumbnailWidth, final int thumbnailHeight) {
|
||||||
slideDeck.clear();
|
slideDeck.clear();
|
||||||
slideDeck.addSlide(slide);
|
slideDeck.addSlide(slide);
|
||||||
thumbnail.setImageDrawable(slide.getThumbnail(thumbnailWidth, thumbnailHeight));
|
new AsyncTask<Void,Void,Drawable>() {
|
||||||
attachmentView.setVisibility(View.VISIBLE);
|
|
||||||
attachmentListener.onAttachmentChanged();
|
@Override
|
||||||
|
protected Drawable doInBackground(Void... params) {
|
||||||
|
return slide.getThumbnail(thumbnailWidth, thumbnailHeight);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onPostExecute(Drawable drawable) {
|
||||||
|
thumbnail.setImageDrawable(drawable);
|
||||||
|
attachmentView.setVisibility(View.VISIBLE);
|
||||||
|
attachmentListener.onAttachmentChanged();
|
||||||
|
}
|
||||||
|
}.execute();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setMedia(Slide slide) {
|
public void setMedia(Slide slide) {
|
||||||
|
@@ -17,6 +17,8 @@
|
|||||||
package org.thoughtcrime.securesms.mms;
|
package org.thoughtcrime.securesms.mms;
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
|
import android.graphics.Bitmap;
|
||||||
|
import android.graphics.BitmapFactory;
|
||||||
import android.graphics.Color;
|
import android.graphics.Color;
|
||||||
import android.graphics.drawable.AnimationDrawable;
|
import android.graphics.drawable.AnimationDrawable;
|
||||||
import android.graphics.drawable.BitmapDrawable;
|
import android.graphics.drawable.BitmapDrawable;
|
||||||
@@ -41,7 +43,6 @@ import org.thoughtcrime.securesms.crypto.MasterSecret;
|
|||||||
|
|
||||||
import java.io.FileNotFoundException;
|
import java.io.FileNotFoundException;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
|
||||||
import java.lang.ref.SoftReference;
|
import java.lang.ref.SoftReference;
|
||||||
import java.lang.ref.WeakReference;
|
import java.lang.ref.WeakReference;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
@@ -51,6 +52,7 @@ import ws.com.google.android.mms.ContentType;
|
|||||||
import ws.com.google.android.mms.pdu.PduPart;
|
import ws.com.google.android.mms.pdu.PduPart;
|
||||||
|
|
||||||
public class ImageSlide extends Slide {
|
public class ImageSlide extends Slide {
|
||||||
|
private static final String TAG = ImageSlide.class.getSimpleName();
|
||||||
|
|
||||||
private static final int MAX_CACHE_SIZE = 10;
|
private static final int MAX_CACHE_SIZE = 10;
|
||||||
private static final Map<Uri, SoftReference<Drawable>> thumbnailCache =
|
private static final Map<Uri, SoftReference<Drawable>> thumbnailCache =
|
||||||
@@ -77,8 +79,15 @@ public class ImageSlide extends Slide {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
thumbnail = new BitmapDrawable(context.getResources(),
|
Bitmap thumbnailBitmap;
|
||||||
BitmapUtil.createScaledBitmap(context, masterSecret, getUri(), maxWidth, maxHeight));
|
long startDecode = System.currentTimeMillis();
|
||||||
|
Log.w(TAG, (part.getThumbnailUri() == null ? "generating" : "fetching pre-generated") + " thumbnail");
|
||||||
|
if (part.getThumbnailUri() != null) thumbnailBitmap = BitmapFactory.decodeStream(PartAuthority.getPartStream(context, masterSecret, part.getThumbnailUri()));
|
||||||
|
else thumbnailBitmap = BitmapUtil.createScaledBitmap(context, masterSecret, getUri(), maxWidth, maxHeight);
|
||||||
|
|
||||||
|
Log.w(TAG, "thumbnail decode/generate time: " + (System.currentTimeMillis() - startDecode) + "ms");
|
||||||
|
|
||||||
|
thumbnail = new BitmapDrawable(context.getResources(), thumbnailBitmap);
|
||||||
thumbnailCache.put(part.getDataUri(), new SoftReference<>(thumbnail));
|
thumbnailCache.put(part.getDataUri(), new SoftReference<>(thumbnail));
|
||||||
|
|
||||||
return thumbnail;
|
return thumbnail;
|
||||||
@@ -115,7 +124,7 @@ public class ImageSlide extends Slide {
|
|||||||
MmsDatabase.slideResolver.execute(new Runnable() {
|
MmsDatabase.slideResolver.execute(new Runnable() {
|
||||||
@Override
|
@Override
|
||||||
public void run() {
|
public void run() {
|
||||||
final Drawable bitmap = getThumbnail(maxWidth, maxHeight);
|
final Drawable bitmap = getThumbnail(maxWidth, maxHeight);
|
||||||
final ImageView destination = weakImageView.get();
|
final ImageView destination = weakImageView.get();
|
||||||
|
|
||||||
if (destination != null && destination.getDrawable() == temporaryDrawable) {
|
if (destination != null && destination.getDrawable() == temporaryDrawable) {
|
||||||
|
@@ -15,16 +15,21 @@ import java.io.InputStream;
|
|||||||
|
|
||||||
public class PartAuthority {
|
public class PartAuthority {
|
||||||
|
|
||||||
private static final String PART_URI_STRING = "content://org.thoughtcrime.securesms/part";
|
private static final String PART_URI_STRING = "content://org.thoughtcrime.securesms/part";
|
||||||
public static final Uri PART_CONTENT_URI = Uri.parse(PART_URI_STRING);
|
private static final String THUMB_URI_STRING = "content://org.thoughtcrime.securesms/thumb";
|
||||||
|
|
||||||
|
public static final Uri PART_CONTENT_URI = Uri.parse(PART_URI_STRING);
|
||||||
|
public static final Uri THUMB_CONTENT_URI = Uri.parse(THUMB_URI_STRING);
|
||||||
|
|
||||||
private static final int PART_ROW = 1;
|
private static final int PART_ROW = 1;
|
||||||
|
private static final int THUMB_ROW = 2;
|
||||||
|
|
||||||
private static final UriMatcher uriMatcher;
|
private static final UriMatcher uriMatcher;
|
||||||
|
|
||||||
static {
|
static {
|
||||||
uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
|
uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
|
||||||
uriMatcher.addURI("org.thoughtcrime.securesms", "part/#", PART_ROW);
|
uriMatcher.addURI("org.thoughtcrime.securesms", "part/#", PART_ROW);
|
||||||
|
uriMatcher.addURI("org.thoughtcrime.securesms", "thumb/#", THUMB_ROW);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static InputStream getPartStream(Context context, MasterSecret masterSecret, Uri uri)
|
public static InputStream getPartStream(Context context, MasterSecret masterSecret, Uri uri)
|
||||||
@@ -35,6 +40,7 @@ public class PartAuthority {
|
|||||||
|
|
||||||
switch (match) {
|
switch (match) {
|
||||||
case PART_ROW: return partDatabase.getPartStream(masterSecret, ContentUris.parseId(uri));
|
case PART_ROW: return partDatabase.getPartStream(masterSecret, ContentUris.parseId(uri));
|
||||||
|
case THUMB_ROW: return partDatabase.getThumbnailStream(masterSecret, ContentUris.parseId(uri));
|
||||||
default: return context.getContentResolver().openInputStream(uri);
|
default: return context.getContentResolver().openInputStream(uri);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
package org.thoughtcrime.securesms.mms;
|
package org.thoughtcrime.securesms.mms;
|
||||||
|
|
||||||
import java.io.FileNotFoundException;
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
|
|
||||||
@@ -25,10 +24,7 @@ import org.w3c.dom.smil.SMILDocument;
|
|||||||
import org.w3c.dom.smil.SMILMediaElement;
|
import org.w3c.dom.smil.SMILMediaElement;
|
||||||
import org.w3c.dom.smil.SMILRegionElement;
|
import org.w3c.dom.smil.SMILRegionElement;
|
||||||
import org.thoughtcrime.securesms.crypto.MasterSecret;
|
import org.thoughtcrime.securesms.crypto.MasterSecret;
|
||||||
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
|
||||||
import org.thoughtcrime.securesms.providers.PartProvider;
|
|
||||||
|
|
||||||
import android.content.ContentUris;
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.graphics.Bitmap;
|
import android.graphics.Bitmap;
|
||||||
import android.graphics.drawable.Drawable;
|
import android.graphics.drawable.Drawable;
|
||||||
@@ -42,10 +38,10 @@ public abstract class Slide {
|
|||||||
|
|
||||||
public static final int MAX_MESSAGE_SIZE = 280 * 1024;
|
public static final int MAX_MESSAGE_SIZE = 280 * 1024;
|
||||||
|
|
||||||
protected final PduPart part;
|
protected final PduPart part;
|
||||||
protected final Context context;
|
protected final Context context;
|
||||||
protected MasterSecret masterSecret;
|
protected MasterSecret masterSecret;
|
||||||
|
|
||||||
public Slide(Context context, PduPart part) {
|
public Slide(Context context, PduPart part) {
|
||||||
this.part = part;
|
this.part = part;
|
||||||
this.context = context;
|
this.context = context;
|
||||||
@@ -56,10 +52,6 @@ public abstract class Slide {
|
|||||||
this.masterSecret = masterSecret;
|
this.masterSecret = masterSecret;
|
||||||
}
|
}
|
||||||
|
|
||||||
public InputStream getPartDataInputStream() throws FileNotFoundException {
|
|
||||||
return PartAuthority.getPartStream(context, masterSecret, part.getDataUri());
|
|
||||||
}
|
|
||||||
|
|
||||||
protected byte[] getPartData() {
|
protected byte[] getPartData() {
|
||||||
try {
|
try {
|
||||||
if (part.getData() != null)
|
if (part.getData() != null)
|
||||||
|
@@ -19,6 +19,7 @@ package org.thoughtcrime.securesms.mms;
|
|||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
|
import android.widget.ImageView;
|
||||||
|
|
||||||
import org.thoughtcrime.securesms.util.SmilUtil;
|
import org.thoughtcrime.securesms.util.SmilUtil;
|
||||||
import org.w3c.dom.smil.SMILDocument;
|
import org.w3c.dom.smil.SMILDocument;
|
||||||
|
@@ -5,12 +5,19 @@ import android.graphics.Bitmap;
|
|||||||
import android.graphics.BitmapFactory;
|
import android.graphics.BitmapFactory;
|
||||||
import android.graphics.Canvas;
|
import android.graphics.Canvas;
|
||||||
import android.graphics.Matrix;
|
import android.graphics.Matrix;
|
||||||
|
import android.graphics.Color;
|
||||||
import android.graphics.Paint;
|
import android.graphics.Paint;
|
||||||
import android.graphics.PorterDuff;
|
import android.graphics.PorterDuff;
|
||||||
|
import android.graphics.PorterDuff.Mode;
|
||||||
import android.graphics.PorterDuffXfermode;
|
import android.graphics.PorterDuffXfermode;
|
||||||
import android.graphics.Rect;
|
import android.graphics.Rect;
|
||||||
|
import android.graphics.RectF;
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
|
import android.util.Pair;
|
||||||
|
|
||||||
|
import org.thoughtcrime.securesms.crypto.MasterSecret;
|
||||||
|
import org.thoughtcrime.securesms.mms.PartAuthority;
|
||||||
|
|
||||||
import java.io.BufferedInputStream;
|
import java.io.BufferedInputStream;
|
||||||
import java.io.ByteArrayOutputStream;
|
import java.io.ByteArrayOutputStream;
|
||||||
@@ -151,11 +158,14 @@ public class BitmapUtil {
|
|||||||
aspectWidth = (aspectHeight / options.outHeight) * options.outWidth;
|
aspectWidth = (aspectHeight / options.outHeight) * options.outWidth;
|
||||||
}
|
}
|
||||||
|
|
||||||
Log.w(TAG, "fine scale " + options.outWidth + "x" + options.outHeight +
|
final int fineWidth = Math.round(aspectWidth);
|
||||||
" => " + aspectWidth + "x" + aspectHeight);
|
final int fineHeight = Math.round(aspectHeight);
|
||||||
|
|
||||||
|
Log.w(TAG, "fine scale " + options.outWidth + "x" + options.outHeight +
|
||||||
|
" => " + fineWidth + "x" + fineHeight);
|
||||||
Bitmap scaledThumbnail = null;
|
Bitmap scaledThumbnail = null;
|
||||||
try {
|
try {
|
||||||
scaledThumbnail = Bitmap.createScaledBitmap(roughThumbnail, (int) aspectWidth, (int) aspectHeight, true);
|
scaledThumbnail = Bitmap.createScaledBitmap(roughThumbnail, fineWidth, fineHeight, true);
|
||||||
} finally {
|
} finally {
|
||||||
if (roughThumbnail != scaledThumbnail) roughThumbnail.recycle();
|
if (roughThumbnail != scaledThumbnail) roughThumbnail.recycle();
|
||||||
}
|
}
|
||||||
|
@@ -19,9 +19,13 @@ package ws.com.google.android.mms.pdu;
|
|||||||
|
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
|
|
||||||
|
import org.thoughtcrime.securesms.util.Util;
|
||||||
|
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
|
import ws.com.google.android.mms.ContentType;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The pdu part.
|
* The pdu part.
|
||||||
*/
|
*/
|
||||||
@@ -121,6 +125,7 @@ public class PduPart {
|
|||||||
|
|
||||||
private static final String TAG = "PduPart";
|
private static final String TAG = "PduPart";
|
||||||
|
|
||||||
|
private Uri thumbnailUri;
|
||||||
private boolean isEncrypted;
|
private boolean isEncrypted;
|
||||||
private boolean isPendingPush;
|
private boolean isPendingPush;
|
||||||
private long dataSize;
|
private long dataSize;
|
||||||
@@ -156,7 +161,15 @@ public class PduPart {
|
|||||||
public boolean isPendingPush() {
|
public boolean isPendingPush() {
|
||||||
return isPendingPush;
|
return isPendingPush;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void setThumbnailUri(Uri thumbnailUri) {
|
||||||
|
this.thumbnailUri = thumbnailUri;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Uri getThumbnailUri() {
|
||||||
|
return this.thumbnailUri;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set part data. The data are stored as byte array.
|
* Set part data. The data are stored as byte array.
|
||||||
*
|
*
|
||||||
@@ -318,7 +331,7 @@ public class PduPart {
|
|||||||
/**
|
/**
|
||||||
* Set Content-Type value.
|
* Set Content-Type value.
|
||||||
*
|
*
|
||||||
* @param value the value
|
* @param contentType the value
|
||||||
* @throws NullPointerException if the value is null.
|
* @throws NullPointerException if the value is null.
|
||||||
*/
|
*/
|
||||||
public void setContentType(byte[] contentType) {
|
public void setContentType(byte[] contentType) {
|
||||||
@@ -341,7 +354,7 @@ public class PduPart {
|
|||||||
/**
|
/**
|
||||||
* Set Content-Transfer-Encoding value
|
* Set Content-Transfer-Encoding value
|
||||||
*
|
*
|
||||||
* @param contentId the content-id value
|
* @param contentTransferEncoding the value
|
||||||
* @throws NullPointerException if the value is null.
|
* @throws NullPointerException if the value is null.
|
||||||
*/
|
*/
|
||||||
public void setContentTransferEncoding(byte[] contentTransferEncoding) {
|
public void setContentTransferEncoding(byte[] contentTransferEncoding) {
|
||||||
|
Reference in New Issue
Block a user