swap out transcoder library
the transcoder library we used hasn’t been updated in years this commit switches to a maintained fork https://natario1.github.io/Transcoder/
This commit is contained in:
parent
3f315751a1
commit
3075833ab3
|
@ -64,7 +64,8 @@ dependencies {
|
||||||
implementation 'org.whispersystems:signal-protocol-java:2.6.2'
|
implementation 'org.whispersystems:signal-protocol-java:2.6.2'
|
||||||
implementation 'com.makeramen:roundedimageview:2.3.0'
|
implementation 'com.makeramen:roundedimageview:2.3.0'
|
||||||
implementation "com.wefika:flowlayout:0.4.1"
|
implementation "com.wefika:flowlayout:0.4.1"
|
||||||
implementation 'net.ypresto.androidtranscoder:android-transcoder:0.3.0'
|
implementation 'com.otaliastudios:transcoder:0.10.3'
|
||||||
|
|
||||||
implementation 'org.jxmpp:jxmpp-jid:1.0.1'
|
implementation 'org.jxmpp:jxmpp-jid:1.0.1'
|
||||||
implementation 'org.osmdroid:osmdroid-android:6.1.10'
|
implementation 'org.osmdroid:osmdroid-android:6.1.10'
|
||||||
implementation 'org.hsluv:hsluv:0.2'
|
implementation 'org.hsluv:hsluv:0.2'
|
||||||
|
|
|
@ -3,16 +3,19 @@ package eu.siacs.conversations.services;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.SharedPreferences;
|
import android.content.SharedPreferences;
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
import android.os.ParcelFileDescriptor;
|
|
||||||
import android.preference.PreferenceManager;
|
import android.preference.PreferenceManager;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
|
|
||||||
import net.ypresto.androidtranscoder.MediaTranscoder;
|
import androidx.annotation.NonNull;
|
||||||
import net.ypresto.androidtranscoder.format.MediaFormatStrategy;
|
|
||||||
|
import com.otaliastudios.transcoder.Transcoder;
|
||||||
|
import com.otaliastudios.transcoder.TranscoderListener;
|
||||||
|
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.FileDescriptor;
|
|
||||||
import java.io.FileNotFoundException;
|
import java.io.FileNotFoundException;
|
||||||
|
import java.util.Objects;
|
||||||
import java.util.concurrent.ExecutionException;
|
import java.util.concurrent.ExecutionException;
|
||||||
import java.util.concurrent.Future;
|
import java.util.concurrent.Future;
|
||||||
|
|
||||||
|
@ -23,11 +26,10 @@ import eu.siacs.conversations.entities.DownloadableFile;
|
||||||
import eu.siacs.conversations.entities.Message;
|
import eu.siacs.conversations.entities.Message;
|
||||||
import eu.siacs.conversations.persistance.FileBackend;
|
import eu.siacs.conversations.persistance.FileBackend;
|
||||||
import eu.siacs.conversations.ui.UiCallback;
|
import eu.siacs.conversations.ui.UiCallback;
|
||||||
import eu.siacs.conversations.utils.Android360pFormatStrategy;
|
|
||||||
import eu.siacs.conversations.utils.Android720pFormatStrategy;
|
|
||||||
import eu.siacs.conversations.utils.MimeUtils;
|
import eu.siacs.conversations.utils.MimeUtils;
|
||||||
|
import eu.siacs.conversations.utils.TranscoderStrategies;
|
||||||
|
|
||||||
public class AttachFileToConversationRunnable implements Runnable, MediaTranscoder.Listener {
|
public class AttachFileToConversationRunnable implements Runnable, TranscoderListener {
|
||||||
|
|
||||||
private final XmppConnectionService mXmppConnectionService;
|
private final XmppConnectionService mXmppConnectionService;
|
||||||
private final Message message;
|
private final Message message;
|
||||||
|
@ -91,14 +93,18 @@ public class AttachFileToConversationRunnable implements Runnable, MediaTranscod
|
||||||
mXmppConnectionService.startForcingForegroundNotification();
|
mXmppConnectionService.startForcingForegroundNotification();
|
||||||
message.setRelativeFilePath(message.getUuid() + ".mp4");
|
message.setRelativeFilePath(message.getUuid() + ".mp4");
|
||||||
final DownloadableFile file = mXmppConnectionService.getFileBackend().getFile(message);
|
final DownloadableFile file = mXmppConnectionService.getFileBackend().getFile(message);
|
||||||
final MediaFormatStrategy formatStrategy = "720".equals(getVideoCompression()) ? new Android720pFormatStrategy() : new Android360pFormatStrategy();
|
if (Objects.requireNonNull(file.getParentFile()).mkdirs()) {
|
||||||
file.getParentFile().mkdirs();
|
Log.d(Config.LOGTAG, "created parent directory for video file");
|
||||||
final ParcelFileDescriptor parcelFileDescriptor = mXmppConnectionService.getContentResolver().openFileDescriptor(uri, "r");
|
|
||||||
if (parcelFileDescriptor == null) {
|
|
||||||
throw new FileNotFoundException("Parcel File Descriptor was null");
|
|
||||||
}
|
}
|
||||||
FileDescriptor fileDescriptor = parcelFileDescriptor.getFileDescriptor();
|
|
||||||
Future<Void> future = MediaTranscoder.getInstance().transcodeVideo(fileDescriptor, file.getAbsolutePath(), formatStrategy, this);
|
final boolean highQuality = "720".equals(getVideoCompression());
|
||||||
|
|
||||||
|
final Future<Void> future = Transcoder.into(file.getAbsolutePath()).
|
||||||
|
addDataSource(mXmppConnectionService, uri)
|
||||||
|
.setVideoTrackStrategy(highQuality ? TranscoderStrategies.VIDEO_720P : TranscoderStrategies.VIDEO_360P)
|
||||||
|
.setAudioTrackStrategy(highQuality ? TranscoderStrategies.AUDIO_HQ : TranscoderStrategies.AUDIO_MQ)
|
||||||
|
.setListener(this)
|
||||||
|
.transcode();
|
||||||
try {
|
try {
|
||||||
future.get();
|
future.get();
|
||||||
} catch (InterruptedException e) {
|
} catch (InterruptedException e) {
|
||||||
|
@ -123,7 +129,7 @@ public class AttachFileToConversationRunnable implements Runnable, MediaTranscod
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onTranscodeCompleted() {
|
public void onTranscodeCompleted(int successCode) {
|
||||||
mXmppConnectionService.stopForcingForegroundNotification();
|
mXmppConnectionService.stopForcingForegroundNotification();
|
||||||
final File file = mXmppConnectionService.getFileBackend().getFile(message);
|
final File file = mXmppConnectionService.getFileBackend().getFile(message);
|
||||||
long convertedFileSize = mXmppConnectionService.getFileBackend().getFile(message).getSize();
|
long convertedFileSize = mXmppConnectionService.getFileBackend().getFile(message).getSize();
|
||||||
|
@ -153,9 +159,9 @@ public class AttachFileToConversationRunnable implements Runnable, MediaTranscod
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onTranscodeFailed(Exception e) {
|
public void onTranscodeFailed(@NonNull @NotNull Throwable exception) {
|
||||||
mXmppConnectionService.stopForcingForegroundNotification();
|
mXmppConnectionService.stopForcingForegroundNotification();
|
||||||
Log.d(Config.LOGTAG,"video transcoding failed",e);
|
Log.d(Config.LOGTAG, "video transcoding failed", exception);
|
||||||
processAsFile();
|
processAsFile();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,76 +0,0 @@
|
||||||
package eu.siacs.conversations.utils;
|
|
||||||
|
|
||||||
import android.media.MediaCodecInfo;
|
|
||||||
import android.media.MediaFormat;
|
|
||||||
import android.os.Build;
|
|
||||||
import android.util.Log;
|
|
||||||
|
|
||||||
import androidx.annotation.RequiresApi;
|
|
||||||
|
|
||||||
import net.ypresto.androidtranscoder.format.MediaFormatExtraConstants;
|
|
||||||
import net.ypresto.androidtranscoder.format.MediaFormatStrategy;
|
|
||||||
import net.ypresto.androidtranscoder.format.OutputFormatUnavailableException;
|
|
||||||
|
|
||||||
import eu.siacs.conversations.Config;
|
|
||||||
|
|
||||||
public class Android360pFormatStrategy implements MediaFormatStrategy {
|
|
||||||
|
|
||||||
private static final int LONGER_LENGTH = 640;
|
|
||||||
private static final int SHORTER_LENGTH = 360;
|
|
||||||
private static final int DEFAULT_VIDEO_BITRATE = 1000 * 1000;
|
|
||||||
private static final int DEFAULT_AUDIO_BITRATE = 128 * 1000;
|
|
||||||
private final int mVideoBitrate;
|
|
||||||
private final int mAudioBitrate;
|
|
||||||
private final int mAudioChannels;
|
|
||||||
|
|
||||||
public Android360pFormatStrategy() {
|
|
||||||
mVideoBitrate = DEFAULT_VIDEO_BITRATE;
|
|
||||||
mAudioBitrate = DEFAULT_AUDIO_BITRATE;
|
|
||||||
mAudioChannels = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
@RequiresApi(api = Build.VERSION_CODES.JELLY_BEAN_MR2)
|
|
||||||
@Override
|
|
||||||
public MediaFormat createVideoOutputFormat(MediaFormat inputFormat) {
|
|
||||||
int width = inputFormat.getInteger(MediaFormat.KEY_WIDTH);
|
|
||||||
int height = inputFormat.getInteger(MediaFormat.KEY_HEIGHT);
|
|
||||||
int longer, shorter, outWidth, outHeight;
|
|
||||||
if (width >= height) {
|
|
||||||
longer = width;
|
|
||||||
shorter = height;
|
|
||||||
outWidth = LONGER_LENGTH;
|
|
||||||
outHeight = SHORTER_LENGTH;
|
|
||||||
} else {
|
|
||||||
shorter = width;
|
|
||||||
longer = height;
|
|
||||||
outWidth = SHORTER_LENGTH;
|
|
||||||
outHeight = LONGER_LENGTH;
|
|
||||||
}
|
|
||||||
if (longer * 9 != shorter * 16) {
|
|
||||||
throw new OutputFormatUnavailableException("This video is not 16:9, and is not able to transcode. (" + width + "x" + height + ")");
|
|
||||||
}
|
|
||||||
if (shorter <= SHORTER_LENGTH) {
|
|
||||||
Log.d(Config.LOGTAG, "This video is less or equal to 360p, pass-through. (" + width + "x" + height + ")");
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
MediaFormat format = MediaFormat.createVideoFormat("video/avc", outWidth, outHeight);
|
|
||||||
format.setInteger(MediaFormat.KEY_BIT_RATE, mVideoBitrate);
|
|
||||||
format.setInteger(MediaFormat.KEY_FRAME_RATE, 30);
|
|
||||||
format.setInteger(MediaFormat.KEY_I_FRAME_INTERVAL, 3);
|
|
||||||
format.setInteger(MediaFormat.KEY_COLOR_FORMAT, MediaCodecInfo.CodecCapabilities.COLOR_FormatSurface);
|
|
||||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
|
|
||||||
format.setInteger(MediaFormat.KEY_PROFILE ,MediaCodecInfo.CodecProfileLevel.AVCProfileBaseline);
|
|
||||||
format.setInteger(MediaFormat.KEY_LEVEL, MediaCodecInfo.CodecProfileLevel.AVCLevel13);
|
|
||||||
}
|
|
||||||
return format;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public MediaFormat createAudioOutputFormat(MediaFormat inputFormat) {
|
|
||||||
final MediaFormat format = MediaFormat.createAudioFormat(MediaFormatExtraConstants.MIMETYPE_AUDIO_AAC, inputFormat.getInteger(MediaFormat.KEY_SAMPLE_RATE), mAudioChannels);
|
|
||||||
format.setInteger(MediaFormat.KEY_AAC_PROFILE, MediaCodecInfo.CodecProfileLevel.AACObjectLC);
|
|
||||||
format.setInteger(MediaFormat.KEY_BIT_RATE, mAudioBitrate);
|
|
||||||
return format;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,76 +0,0 @@
|
||||||
package eu.siacs.conversations.utils;
|
|
||||||
|
|
||||||
import android.media.MediaCodecInfo;
|
|
||||||
import android.media.MediaFormat;
|
|
||||||
import android.os.Build;
|
|
||||||
import android.util.Log;
|
|
||||||
|
|
||||||
import androidx.annotation.RequiresApi;
|
|
||||||
|
|
||||||
import net.ypresto.androidtranscoder.format.MediaFormatExtraConstants;
|
|
||||||
import net.ypresto.androidtranscoder.format.MediaFormatStrategy;
|
|
||||||
import net.ypresto.androidtranscoder.format.OutputFormatUnavailableException;
|
|
||||||
|
|
||||||
import eu.siacs.conversations.Config;
|
|
||||||
|
|
||||||
public class Android720pFormatStrategy implements MediaFormatStrategy {
|
|
||||||
|
|
||||||
private static final int LONGER_LENGTH = 1280;
|
|
||||||
private static final int SHORTER_LENGTH = 720;
|
|
||||||
private static final int DEFAULT_VIDEO_BITRATE = 2000 * 1000;
|
|
||||||
private static final int DEFAULT_AUDIO_BITRATE = 192 * 1000;
|
|
||||||
private final int mVideoBitrate;
|
|
||||||
private final int mAudioBitrate;
|
|
||||||
private final int mAudioChannels;
|
|
||||||
|
|
||||||
public Android720pFormatStrategy() {
|
|
||||||
mVideoBitrate = DEFAULT_VIDEO_BITRATE;
|
|
||||||
mAudioBitrate = DEFAULT_AUDIO_BITRATE;
|
|
||||||
mAudioChannels = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
@RequiresApi(api = Build.VERSION_CODES.JELLY_BEAN_MR2)
|
|
||||||
@Override
|
|
||||||
public MediaFormat createVideoOutputFormat(MediaFormat inputFormat) {
|
|
||||||
int width = inputFormat.getInteger(MediaFormat.KEY_WIDTH);
|
|
||||||
int height = inputFormat.getInteger(MediaFormat.KEY_HEIGHT);
|
|
||||||
int longer, shorter, outWidth, outHeight;
|
|
||||||
if (width >= height) {
|
|
||||||
longer = width;
|
|
||||||
shorter = height;
|
|
||||||
outWidth = LONGER_LENGTH;
|
|
||||||
outHeight = SHORTER_LENGTH;
|
|
||||||
} else {
|
|
||||||
shorter = width;
|
|
||||||
longer = height;
|
|
||||||
outWidth = SHORTER_LENGTH;
|
|
||||||
outHeight = LONGER_LENGTH;
|
|
||||||
}
|
|
||||||
if (longer * 9 != shorter * 16) {
|
|
||||||
throw new OutputFormatUnavailableException("This video is not 16:9, and is not able to transcode. (" + width + "x" + height + ")");
|
|
||||||
}
|
|
||||||
if (shorter <= SHORTER_LENGTH) {
|
|
||||||
Log.d(Config.LOGTAG, "This video is less or equal to 720p, pass-through. (" + width + "x" + height + ")");
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
MediaFormat format = MediaFormat.createVideoFormat("video/avc", outWidth, outHeight);
|
|
||||||
format.setInteger(MediaFormat.KEY_BIT_RATE, mVideoBitrate);
|
|
||||||
format.setInteger(MediaFormat.KEY_FRAME_RATE, 30);
|
|
||||||
format.setInteger(MediaFormat.KEY_I_FRAME_INTERVAL, 3);
|
|
||||||
format.setInteger(MediaFormat.KEY_COLOR_FORMAT, MediaCodecInfo.CodecCapabilities.COLOR_FormatSurface);
|
|
||||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
|
|
||||||
format.setInteger(MediaFormat.KEY_PROFILE ,MediaCodecInfo.CodecProfileLevel.AVCProfileBaseline);
|
|
||||||
format.setInteger(MediaFormat.KEY_LEVEL, MediaCodecInfo.CodecProfileLevel.AVCLevel13);
|
|
||||||
}
|
|
||||||
return format;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public MediaFormat createAudioOutputFormat(MediaFormat inputFormat) {
|
|
||||||
final MediaFormat format = MediaFormat.createAudioFormat(MediaFormatExtraConstants.MIMETYPE_AUDIO_AAC, inputFormat.getInteger(MediaFormat.KEY_SAMPLE_RATE), mAudioChannels);
|
|
||||||
format.setInteger(MediaFormat.KEY_AAC_PROFILE, MediaCodecInfo.CodecProfileLevel.AACObjectLC);
|
|
||||||
format.setInteger(MediaFormat.KEY_BIT_RATE, mAudioBitrate);
|
|
||||||
return format;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -0,0 +1,41 @@
|
||||||
|
package eu.siacs.conversations.utils;
|
||||||
|
|
||||||
|
import com.otaliastudios.transcoder.strategy.DefaultAudioStrategy;
|
||||||
|
import com.otaliastudios.transcoder.strategy.DefaultVideoStrategy;
|
||||||
|
|
||||||
|
public final class TranscoderStrategies {
|
||||||
|
|
||||||
|
public static final DefaultVideoStrategy VIDEO_720P = DefaultVideoStrategy.atMost(720)
|
||||||
|
.bitRate(2L * 1000 * 1000)
|
||||||
|
.frameRate(30)
|
||||||
|
.keyFrameInterval(3F)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
public static final DefaultVideoStrategy VIDEO_360P = DefaultVideoStrategy.atMost(360)
|
||||||
|
.bitRate(1000 * 1000)
|
||||||
|
.frameRate(30)
|
||||||
|
.keyFrameInterval(3F)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
//TODO do we want to add 240p (@500kbs) and 1080p (@4mbs?) ?
|
||||||
|
// see suggested bit rates on https://www.videoproc.com/media-converter/bitrate-setting-for-h264.htm
|
||||||
|
|
||||||
|
public static final DefaultAudioStrategy AUDIO_HQ = DefaultAudioStrategy.builder()
|
||||||
|
.bitRate(192 * 1000)
|
||||||
|
.channels(2)
|
||||||
|
.sampleRate(DefaultAudioStrategy.SAMPLE_RATE_AS_INPUT)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
public static final DefaultAudioStrategy AUDIO_MQ = DefaultAudioStrategy.builder()
|
||||||
|
.bitRate(128 * 1000)
|
||||||
|
.channels(2)
|
||||||
|
.sampleRate(DefaultAudioStrategy.SAMPLE_RATE_AS_INPUT)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
//TODO if we add 144p we definitely want to add a lower audio bit rate as well
|
||||||
|
|
||||||
|
private TranscoderStrategies() {
|
||||||
|
throw new IllegalStateException("Do not instantiate me");
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue