- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我想在同一个 SurfaceView 中播放几张图片后播放一段视频。如果我在视频后显示图像,一切正常。但是如果我想在图像之后播放视频,SurfaceView 将保持空白(黑色)。谁能帮忙?好像是清除或重置SurfaceView的问题!
SurfaceView的代码(自定义SurfaceView!!!):
public class MySurfaceView extends SurfaceView implements SurfaceHolder.Callback, MediaPlayer.OnCompletionListener {
private Bitmap image;
private MediaPlayer mediaPlayer;
private List<MySurfaceListener> objectsToCall = new ArrayList<>();
public MySurfaceView(Context context) {
super(context);
getHolder().addCallback(this);
}
public void addOnPlayingFinishedListener(MySurfaceListener listener) {
objectsToCall.add(listener);
}
public void notifyAllOnPlayingFinishedListener() {
for(MySurfaceListener l : objectsToCall) {
l.onMediaPlayFinished();
}
}
public static int calculateInSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) {
// Helper method to scale down the image files
}
public static Bitmap decodeSampledBitmapFromFile(String path, int reqWidth, int reqHeight) {
// Another helper method for the image files
}
public void drawImage(String path, final int durationMs) {
this.image = decodeSampledBitmapFromFile(path, 1920, 1080);
Canvas canvas = null;
try {
canvas = getHolder().lockCanvas(null);
synchronized (getHolder()) {
onDraw(canvas);
}
} catch (Exception e) {
e.printStackTrace();
} finally {
if (canvas != null) {
getHolder().unlockCanvasAndPost(canvas);
}
image.recycle();
image = null;
}
Thread showPictureTimer = new Thread(new Runnable() {
@Override
public void run() {
try {
Thread.sleep(durationMs);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
});
showPictureTimer.start();
try {
showPictureTimer.join();
} catch (InterruptedException e) {
e.printStackTrace();
}
notifyAllOnPlayingFinishedListener();
}
public void playVideo(String path) {
try {
mediaPlayer = new MediaPlayer();
mediaPlayer.setOnCompletionListener(this);
mediaPlayer.setDisplay(getHolder());
mediaPlayer.setDataSource(path);
mediaPlayer.prepare();
mediaPlayer.start();
} catch (IllegalArgumentException e) {
Log.d("MyMediaPlayerControl", e.getMessage());
} catch (IllegalStateException e) {
Log.d("MyMediaPlayerControl", e.getMessage());
} catch (IOException e) {
Log.d("MyMediaPlayerControl", e.getMessage());
}
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
canvas.drawColor(Color.BLACK);
canvas.drawBitmap(this.image, 0, 0, new Paint());
}
public void surfaceChanged(SurfaceHolder arg0, int arg1, int arg2, int arg3) {
Log.e("myApp", "surfaceChanged");
}
public void surfaceCreated(SurfaceHolder arg0) {
Log.e("myApp", "surfaceCreated");
}
public void surfaceDestroyed(SurfaceHolder arg0) {
Log.e("myApp", "surfaceDestroyed");
}
@Override
public void onCompletion(MediaPlayer mp) {
if (mediaPlayer != null) {
mediaPlayer.stop();
mediaPlayer.release();
mediaPlayer = null;
}
notifyAllOnPlayingFinishedListener();
}
}
以及将文件路径提供给用于播放的表面 View 的线程(应该不是问题所在):
public class MediaControlThread extends Thread implements MySurfaceListener {
public static List<String> mediaList = new ArrayList<String>();
MyMediaActivity activity;
private MySurfaceView surfaceView;
private int currentVideo = 0;
public MediaControlThread(MyMediaActivity activity) {
this.activity = activity;
this.surfaceView = activity.getSurfaceView();
this.surfaceView.addOnPlayingFinishedListener(this);
}
@Override
public void run() {
try {
while(mediaList.size() == 0) {
Thread.sleep(1000);
}
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
}
playMedia(mediaList.get(currentVideo));
}
private void playMedia(String mediaPath) {
if(checkFileForImage(mediaPath)) {
surfaceView.drawImage(mediaPath, 2000);
} else {
surfaceView.playVideo(mediaPath);
}
}
private boolean checkFileForImage(String fileName) {
String file = fileName.toLowerCase();
return file.endsWith(".jpg") ||
file.endsWith(".img") ||
file.endsWith(".bmp") ||
file.endsWith(".jpeg") ||
file.endsWith(".ico") ||
file.endsWith(".tif");
}
@Override
public void onMediaPlayFinished() {
surfaceView.invalidate();
surfaceView.destroyDrawingCache();
currentVideo++;
if (currentVideo > mediaList.size() - 1) {
currentVideo = 0;
}
playMedia(mediaList.get(currentVideo));
}
}
最佳答案
不幸的是,这是预期的行为。由于 Android 中的一个怪癖,一旦您使用 Canvas 在 Surface 上绘图,您将无法执行任何其他操作。
你有三个选择:
关于安卓表面 View : Show Video after Images,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33372873/
我想将 EditText 始终设置为 LTR,我该怎么做?android:textDirection 似乎确实是答案,我无法让项目使用该指令进行编译 最佳答案 为 editText 使用引力 andr
我希望我的应用在每次打开时都显示登录屏幕。使用 android:clearTaskOnLaunch="true" 一切正常。但是有一个错误,只有在手机关机并且应用程序首先使用小部件启动时才会发生。 I
实际上我的要求是我想将动态文件加载到像图像一样的网页中。视频,音频等,它来自 Assets 或应用程序自己的文件目录,这不是问题。 我尝试使用以下两种方式。 方式一 这是我的 html 文件在 ass
我正在触发一个 DatePickerDialog,在 api 22 (Android 5.1) 之前它一直在工作并显示良好,我在上面设置混合和最大日期(最小 = 当前日期,最大 = 从当前日期开始的
我有一个 ListView ,我在其中将标题 View 添加到该列表。一切都很好,但是当滚动列表 headerview 也随着列表移动时,所以我想避免 headerview 滚动,我的意思是当我列表到
虽然我在android上做过一些app,但我还是一头雾水。是否可以使用 SDK 4.0 中的功能,并在 android 2.1 或更低版本上运行该应用程序? 我尝试了你们提到的方法,但出现错误 - F
您好,我正在开发小型 android 应用程序,我想在其中显示带有一些元素的简单 gridview。它工作正常。唯一的问题是即使有空间,它也总是只显示两列。它平均将屏幕分成 2 列并仅显示两个元素。如
我正在使用 Android 2.3.3 API 构建一个应用程序。我需要识别方向的变化并执行一些操作。所以我在 Android Manifest 中添加了以下内容, android:configCha
我正在尝试在“点击”包含特定 MIME 类型的 nfc 标签时开始一项 Activity 。我制作了一个 mime 类型为“text/plain”的标签,并将其添加到 list 中:
我可以将一些数据保存到文件中 val byteArrayOutputStream = ByteArrayOutputStream() byteArrayOutputStream.wri
我正在尝试解析一个包含复杂阿拉伯字母的 XML 文件.. 当我在 android 2.3.7 上测试时,并不是所有的阿拉伯字母都被支持,仍然有一些复杂的显示为方 block .. 但是在 androi
我需要编写一个方法来确定设备是平板电脑还是手机。我不需要根据这个显示不同的用户界面。我只需要有关设备的信息,以便将来我可以将其发送到指标。 在互联网上,我找到了很多方法来确定设备是否是平板电脑。我已经
我正在玩文字转语音,让我的测试应用程序更有趣。它适用于模拟器,但不适用于我的手机,因为我的默认语言环境不是英语。 但是,文本是英文的,所以 tts 当然应该使用英文。据我所知,我可以实现一个自动安装,
我正在使用 MVVM 和整洁的架构编写应用程序。在其中一个屏幕上,我需要使用 pagination 实现 RecyclerView。我将使用库 Paging3。 Android 开发者推荐在存储库层使
这个问题在这里已经有了答案: I lost my .keystore file? (12 个回答) 4年前关闭。 当我以前的操作系统损坏并安装新的(7 月 3 日)时,以前的 android_key_
我正在使用 MVVM 和整洁的架构编写应用程序。在其中一个屏幕上,我需要使用 pagination 实现 RecyclerView。我将使用库 Paging3。 Android 开发者推荐在存储库层使
在我的 v27\style.xml 中,我有以下代码来设置白色导航栏: @color/colorPrimary @color/colorPrimaryDark @color/
我想通过发送电子邮件 startActivity(Intent.createChooser(new Intent(android.content.Intent.ACTION_SEND))) 我知道要将
我实现了一个自定义 ListView ,它看起来像 Twitter 时间线。 adapter = new MyClickableListAdapter(this, R.layout.timeline,
我有一个显示启动画面的自定义对话框; mSplashDialog = new Dialog(MyActivity.this,R.layout.splash); mSplashDialog.setCon
我是一名优秀的程序员,十分优秀!