Move rendering code in SurfaceViewRenderer to a separate class.
The new SurfaceEglRenderer helper class extends EglRenderer and implements rendering on a SurfaceView. Bug: webrtc:8242 Change-Id: Ic532fe487755d3b54c6bd03f239d714e1ecb10ad Reviewed-on: https://webrtc-review.googlesource.com/2940 Commit-Queue: Sami Kalliomäki <sakal@webrtc.org> Reviewed-by: Sami Kalliomäki <sakal@webrtc.org> Cr-Commit-Position: refs/heads/master@{#20562}
This commit is contained in:
1
AUTHORS
1
AUTHORS
@ -53,6 +53,7 @@ Vladimir Beloborodov <VladimirTechMan@gmail.com>
|
|||||||
Vicken Simonian <vsimon@gmail.com>
|
Vicken Simonian <vsimon@gmail.com>
|
||||||
Victor Costan <costan@gmail.com>
|
Victor Costan <costan@gmail.com>
|
||||||
Xiaohong Xu <freemine@yeah.net>
|
Xiaohong Xu <freemine@yeah.net>
|
||||||
|
Xiaolei Yu <dreifachstein@gmail.com>
|
||||||
Hans Knoechel <hans@hans-knoechel.de>
|
Hans Knoechel <hans@hans-knoechel.de>
|
||||||
Korniltsev Anatoly <korniltsev.anatoly@gmail.com>
|
Korniltsev Anatoly <korniltsev.anatoly@gmail.com>
|
||||||
|
|
||||||
|
@ -438,6 +438,7 @@ android_library("libjingle_peerconnection_java") {
|
|||||||
"api/org/webrtc/StatsObserver.java",
|
"api/org/webrtc/StatsObserver.java",
|
||||||
"api/org/webrtc/StatsReport.java",
|
"api/org/webrtc/StatsReport.java",
|
||||||
"api/org/webrtc/SurfaceTextureHelper.java",
|
"api/org/webrtc/SurfaceTextureHelper.java",
|
||||||
|
"api/org/webrtc/SurfaceEglRenderer.java",
|
||||||
"api/org/webrtc/SurfaceViewRenderer.java",
|
"api/org/webrtc/SurfaceViewRenderer.java",
|
||||||
"api/org/webrtc/TurnCustomizer.java",
|
"api/org/webrtc/TurnCustomizer.java",
|
||||||
"api/org/webrtc/VideoCapturer.java",
|
"api/org/webrtc/VideoCapturer.java",
|
||||||
|
@ -76,7 +76,7 @@ public class EglRenderer implements VideoRenderer.Callbacks, VideoSink {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private final String name;
|
protected final String name;
|
||||||
|
|
||||||
// |renderThreadHandler| is a handler for communicating with |renderThread|, and is synchronized
|
// |renderThreadHandler| is a handler for communicating with |renderThread|, and is synchronized
|
||||||
// on |handlerLock|.
|
// on |handlerLock|.
|
||||||
|
194
sdk/android/api/org/webrtc/SurfaceEglRenderer.java
Normal file
194
sdk/android/api/org/webrtc/SurfaceEglRenderer.java
Normal file
@ -0,0 +1,194 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2017 The WebRTC project authors. All Rights Reserved.
|
||||||
|
*
|
||||||
|
* Use of this source code is governed by a BSD-style license
|
||||||
|
* that can be found in the LICENSE file in the root of the source
|
||||||
|
* tree. An additional intellectual property rights grant can be found
|
||||||
|
* in the file PATENTS. All contributing project authors may
|
||||||
|
* be found in the AUTHORS file in the root of the source tree.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.webrtc;
|
||||||
|
|
||||||
|
import android.view.SurfaceHolder;
|
||||||
|
import java.util.concurrent.CountDownLatch;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Display the video stream on a Surface.
|
||||||
|
* renderFrame() is asynchronous to avoid blocking the calling thread.
|
||||||
|
* This class is thread safe and handles access from potentially three different threads:
|
||||||
|
* Interaction from the main app in init, release and setMirror.
|
||||||
|
* Interaction from C++ rtc::VideoSinkInterface in renderFrame.
|
||||||
|
* Interaction from SurfaceHolder lifecycle in surfaceCreated, surfaceChanged, and surfaceDestroyed.
|
||||||
|
*/
|
||||||
|
public class SurfaceEglRenderer extends EglRenderer implements SurfaceHolder.Callback {
|
||||||
|
private static final String TAG = "SurfaceEglRenderer";
|
||||||
|
|
||||||
|
// Callback for reporting renderer events. Read-only after initilization so no lock required.
|
||||||
|
private RendererCommon.RendererEvents rendererEvents;
|
||||||
|
|
||||||
|
private final Object layoutLock = new Object();
|
||||||
|
private boolean isRenderingPaused = false;
|
||||||
|
private boolean isFirstFrameRendered;
|
||||||
|
private int rotatedFrameWidth;
|
||||||
|
private int rotatedFrameHeight;
|
||||||
|
private int frameRotation;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* In order to render something, you must first call init().
|
||||||
|
*/
|
||||||
|
public SurfaceEglRenderer(String name) {
|
||||||
|
super(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initialize this class, sharing resources with |sharedContext|. The custom |drawer| will be used
|
||||||
|
* for drawing frames on the EGLSurface. This class is responsible for calling release() on
|
||||||
|
* |drawer|. It is allowed to call init() to reinitialize the renderer after a previous
|
||||||
|
* init()/release() cycle.
|
||||||
|
*/
|
||||||
|
public void init(final EglBase.Context sharedContext,
|
||||||
|
RendererCommon.RendererEvents rendererEvents, final int[] configAttributes,
|
||||||
|
RendererCommon.GlDrawer drawer) {
|
||||||
|
ThreadUtils.checkIsOnMainThread();
|
||||||
|
this.rendererEvents = rendererEvents;
|
||||||
|
synchronized (layoutLock) {
|
||||||
|
isFirstFrameRendered = false;
|
||||||
|
rotatedFrameWidth = 0;
|
||||||
|
rotatedFrameHeight = 0;
|
||||||
|
frameRotation = 0;
|
||||||
|
}
|
||||||
|
super.init(sharedContext, configAttributes, drawer);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void init(final EglBase.Context sharedContext, final int[] configAttributes,
|
||||||
|
RendererCommon.GlDrawer drawer) {
|
||||||
|
init(sharedContext, null /* rendererEvents */, configAttributes, drawer);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Limit render framerate.
|
||||||
|
*
|
||||||
|
* @param fps Limit render framerate to this value, or use Float.POSITIVE_INFINITY to disable fps
|
||||||
|
* reduction.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void setFpsReduction(float fps) {
|
||||||
|
synchronized (layoutLock) {
|
||||||
|
isRenderingPaused = fps == 0f;
|
||||||
|
}
|
||||||
|
super.setFpsReduction(fps);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void disableFpsReduction() {
|
||||||
|
synchronized (layoutLock) {
|
||||||
|
isRenderingPaused = false;
|
||||||
|
}
|
||||||
|
super.disableFpsReduction();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void pauseVideo() {
|
||||||
|
synchronized (layoutLock) {
|
||||||
|
isRenderingPaused = true;
|
||||||
|
}
|
||||||
|
super.pauseVideo();
|
||||||
|
}
|
||||||
|
|
||||||
|
// VideoRenderer.Callbacks interface.
|
||||||
|
@Override
|
||||||
|
public void renderFrame(VideoRenderer.I420Frame frame) {
|
||||||
|
updateFrameDimensionsAndReportEvents(frame);
|
||||||
|
super.renderFrame(frame);
|
||||||
|
}
|
||||||
|
|
||||||
|
// VideoSink interface.
|
||||||
|
@Override
|
||||||
|
public void onFrame(VideoFrame frame) {
|
||||||
|
updateFrameDimensionsAndReportEvents(frame);
|
||||||
|
super.onFrame(frame);
|
||||||
|
}
|
||||||
|
|
||||||
|
// SurfaceHolder.Callback interface.
|
||||||
|
@Override
|
||||||
|
public void surfaceCreated(final SurfaceHolder holder) {
|
||||||
|
ThreadUtils.checkIsOnMainThread();
|
||||||
|
createEglSurface(holder.getSurface());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void surfaceDestroyed(SurfaceHolder holder) {
|
||||||
|
ThreadUtils.checkIsOnMainThread();
|
||||||
|
final CountDownLatch completionLatch = new CountDownLatch(1);
|
||||||
|
releaseEglSurface(completionLatch::countDown);
|
||||||
|
ThreadUtils.awaitUninterruptibly(completionLatch);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
|
||||||
|
ThreadUtils.checkIsOnMainThread();
|
||||||
|
logD("surfaceChanged: format: " + format + " size: " + width + "x" + height);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update frame dimensions and report any changes to |rendererEvents|.
|
||||||
|
private void updateFrameDimensionsAndReportEvents(VideoRenderer.I420Frame frame) {
|
||||||
|
synchronized (layoutLock) {
|
||||||
|
if (isRenderingPaused) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!isFirstFrameRendered) {
|
||||||
|
isFirstFrameRendered = true;
|
||||||
|
logD("Reporting first rendered frame.");
|
||||||
|
if (rendererEvents != null) {
|
||||||
|
rendererEvents.onFirstFrameRendered();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (rotatedFrameWidth != frame.rotatedWidth() || rotatedFrameHeight != frame.rotatedHeight()
|
||||||
|
|| frameRotation != frame.rotationDegree) {
|
||||||
|
logD("Reporting frame resolution changed to " + frame.width + "x" + frame.height
|
||||||
|
+ " with rotation " + frame.rotationDegree);
|
||||||
|
if (rendererEvents != null) {
|
||||||
|
rendererEvents.onFrameResolutionChanged(frame.width, frame.height, frame.rotationDegree);
|
||||||
|
}
|
||||||
|
rotatedFrameWidth = frame.rotatedWidth();
|
||||||
|
rotatedFrameHeight = frame.rotatedHeight();
|
||||||
|
frameRotation = frame.rotationDegree;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update frame dimensions and report any changes to |rendererEvents|.
|
||||||
|
private void updateFrameDimensionsAndReportEvents(VideoFrame frame) {
|
||||||
|
synchronized (layoutLock) {
|
||||||
|
if (isRenderingPaused) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!isFirstFrameRendered) {
|
||||||
|
isFirstFrameRendered = true;
|
||||||
|
logD("Reporting first rendered frame.");
|
||||||
|
if (rendererEvents != null) {
|
||||||
|
rendererEvents.onFirstFrameRendered();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (rotatedFrameWidth != frame.getRotatedWidth()
|
||||||
|
|| rotatedFrameHeight != frame.getRotatedHeight()
|
||||||
|
|| frameRotation != frame.getRotation()) {
|
||||||
|
logD("Reporting frame resolution changed to " + frame.getBuffer().getWidth() + "x"
|
||||||
|
+ frame.getBuffer().getHeight() + " with rotation " + frame.getRotation());
|
||||||
|
if (rendererEvents != null) {
|
||||||
|
rendererEvents.onFrameResolutionChanged(
|
||||||
|
frame.getBuffer().getWidth(), frame.getBuffer().getHeight(), frame.getRotation());
|
||||||
|
}
|
||||||
|
rotatedFrameWidth = frame.getRotatedWidth();
|
||||||
|
rotatedFrameHeight = frame.getRotatedHeight();
|
||||||
|
frameRotation = frame.getRotation();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void logD(String string) {
|
||||||
|
Logging.d(TAG, name + ": " + string);
|
||||||
|
}
|
||||||
|
}
|
@ -13,41 +13,31 @@ package org.webrtc;
|
|||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.res.Resources.NotFoundException;
|
import android.content.res.Resources.NotFoundException;
|
||||||
import android.graphics.Point;
|
import android.graphics.Point;
|
||||||
|
import android.os.Looper;
|
||||||
import android.util.AttributeSet;
|
import android.util.AttributeSet;
|
||||||
import android.view.SurfaceHolder;
|
import android.view.SurfaceHolder;
|
||||||
import android.view.SurfaceView;
|
import android.view.SurfaceView;
|
||||||
import java.util.concurrent.CountDownLatch;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Implements org.webrtc.VideoRenderer.Callbacks by displaying the video stream on a SurfaceView.
|
* Display the video stream on a SurfaceView.
|
||||||
* renderFrame() is asynchronous to avoid blocking the calling thread.
|
|
||||||
* This class is thread safe and handles access from potentially four different threads:
|
|
||||||
* Interaction from the main app in init, release, setMirror, and setScalingtype.
|
|
||||||
* Interaction from C++ rtc::VideoSinkInterface in renderFrame.
|
|
||||||
* Interaction from the Activity lifecycle in surfaceCreated, surfaceChanged, and surfaceDestroyed.
|
|
||||||
* Interaction with the layout framework in onMeasure and onSizeChanged.
|
|
||||||
*/
|
*/
|
||||||
public class SurfaceViewRenderer
|
public class SurfaceViewRenderer extends SurfaceView implements SurfaceHolder.Callback,
|
||||||
extends SurfaceView implements SurfaceHolder.Callback, VideoRenderer.Callbacks, VideoSink {
|
VideoRenderer.Callbacks, VideoSink,
|
||||||
|
RendererCommon.RendererEvents {
|
||||||
private static final String TAG = "SurfaceViewRenderer";
|
private static final String TAG = "SurfaceViewRenderer";
|
||||||
|
|
||||||
// Cached resource name.
|
// Cached resource name.
|
||||||
private final String resourceName;
|
private final String resourceName;
|
||||||
private final RendererCommon.VideoLayoutMeasure videoLayoutMeasure =
|
private final RendererCommon.VideoLayoutMeasure videoLayoutMeasure =
|
||||||
new RendererCommon.VideoLayoutMeasure();
|
new RendererCommon.VideoLayoutMeasure();
|
||||||
private final EglRenderer eglRenderer;
|
private final SurfaceEglRenderer eglRenderer;
|
||||||
|
|
||||||
// Callback for reporting renderer events. Read-only after initilization so no lock required.
|
// Callback for reporting renderer events. Read-only after initilization so no lock required.
|
||||||
private RendererCommon.RendererEvents rendererEvents;
|
private RendererCommon.RendererEvents rendererEvents;
|
||||||
|
|
||||||
private final Object layoutLock = new Object();
|
// Accessed only on the main thread.
|
||||||
private boolean isRenderingPaused = false;
|
|
||||||
private boolean isFirstFrameRendered;
|
|
||||||
private int rotatedFrameWidth;
|
private int rotatedFrameWidth;
|
||||||
private int rotatedFrameHeight;
|
private int rotatedFrameHeight;
|
||||||
private int frameRotation;
|
|
||||||
|
|
||||||
// Accessed only on the main thread.
|
|
||||||
private boolean enableFixedSize;
|
private boolean enableFixedSize;
|
||||||
private int surfaceWidth;
|
private int surfaceWidth;
|
||||||
private int surfaceHeight;
|
private int surfaceHeight;
|
||||||
@ -58,8 +48,9 @@ public class SurfaceViewRenderer
|
|||||||
public SurfaceViewRenderer(Context context) {
|
public SurfaceViewRenderer(Context context) {
|
||||||
super(context);
|
super(context);
|
||||||
this.resourceName = getResourceName();
|
this.resourceName = getResourceName();
|
||||||
eglRenderer = new EglRenderer(resourceName);
|
eglRenderer = new SurfaceEglRenderer(resourceName);
|
||||||
getHolder().addCallback(this);
|
getHolder().addCallback(this);
|
||||||
|
getHolder().addCallback(eglRenderer);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -68,8 +59,9 @@ public class SurfaceViewRenderer
|
|||||||
public SurfaceViewRenderer(Context context, AttributeSet attrs) {
|
public SurfaceViewRenderer(Context context, AttributeSet attrs) {
|
||||||
super(context, attrs);
|
super(context, attrs);
|
||||||
this.resourceName = getResourceName();
|
this.resourceName = getResourceName();
|
||||||
eglRenderer = new EglRenderer(resourceName);
|
eglRenderer = new SurfaceEglRenderer(resourceName);
|
||||||
getHolder().addCallback(this);
|
getHolder().addCallback(this);
|
||||||
|
getHolder().addCallback(eglRenderer);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -91,13 +83,9 @@ public class SurfaceViewRenderer
|
|||||||
RendererCommon.GlDrawer drawer) {
|
RendererCommon.GlDrawer drawer) {
|
||||||
ThreadUtils.checkIsOnMainThread();
|
ThreadUtils.checkIsOnMainThread();
|
||||||
this.rendererEvents = rendererEvents;
|
this.rendererEvents = rendererEvents;
|
||||||
synchronized (layoutLock) {
|
|
||||||
isFirstFrameRendered = false;
|
|
||||||
rotatedFrameWidth = 0;
|
rotatedFrameWidth = 0;
|
||||||
rotatedFrameHeight = 0;
|
rotatedFrameHeight = 0;
|
||||||
frameRotation = 0;
|
eglRenderer.init(sharedContext, this /* rendererEvents */, configAttributes, drawer);
|
||||||
}
|
|
||||||
eglRenderer.init(sharedContext, configAttributes, drawer);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -181,37 +169,26 @@ public class SurfaceViewRenderer
|
|||||||
* reduction.
|
* reduction.
|
||||||
*/
|
*/
|
||||||
public void setFpsReduction(float fps) {
|
public void setFpsReduction(float fps) {
|
||||||
synchronized (layoutLock) {
|
|
||||||
isRenderingPaused = fps == 0f;
|
|
||||||
}
|
|
||||||
eglRenderer.setFpsReduction(fps);
|
eglRenderer.setFpsReduction(fps);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void disableFpsReduction() {
|
public void disableFpsReduction() {
|
||||||
synchronized (layoutLock) {
|
|
||||||
isRenderingPaused = false;
|
|
||||||
}
|
|
||||||
eglRenderer.disableFpsReduction();
|
eglRenderer.disableFpsReduction();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void pauseVideo() {
|
public void pauseVideo() {
|
||||||
synchronized (layoutLock) {
|
|
||||||
isRenderingPaused = true;
|
|
||||||
}
|
|
||||||
eglRenderer.pauseVideo();
|
eglRenderer.pauseVideo();
|
||||||
}
|
}
|
||||||
|
|
||||||
// VideoRenderer.Callbacks interface.
|
// VideoRenderer.Callbacks interface.
|
||||||
@Override
|
@Override
|
||||||
public void renderFrame(VideoRenderer.I420Frame frame) {
|
public void renderFrame(VideoRenderer.I420Frame frame) {
|
||||||
updateFrameDimensionsAndReportEvents(frame);
|
|
||||||
eglRenderer.renderFrame(frame);
|
eglRenderer.renderFrame(frame);
|
||||||
}
|
}
|
||||||
|
|
||||||
// VideoSink interface.
|
// VideoSink interface.
|
||||||
@Override
|
@Override
|
||||||
public void onFrame(VideoFrame frame) {
|
public void onFrame(VideoFrame frame) {
|
||||||
updateFrameDimensionsAndReportEvents(frame);
|
|
||||||
eglRenderer.onFrame(frame);
|
eglRenderer.onFrame(frame);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -219,11 +196,8 @@ public class SurfaceViewRenderer
|
|||||||
@Override
|
@Override
|
||||||
protected void onMeasure(int widthSpec, int heightSpec) {
|
protected void onMeasure(int widthSpec, int heightSpec) {
|
||||||
ThreadUtils.checkIsOnMainThread();
|
ThreadUtils.checkIsOnMainThread();
|
||||||
final Point size;
|
Point size =
|
||||||
synchronized (layoutLock) {
|
|
||||||
size =
|
|
||||||
videoLayoutMeasure.measure(widthSpec, heightSpec, rotatedFrameWidth, rotatedFrameHeight);
|
videoLayoutMeasure.measure(widthSpec, heightSpec, rotatedFrameWidth, rotatedFrameHeight);
|
||||||
}
|
|
||||||
setMeasuredDimension(size.x, size.y);
|
setMeasuredDimension(size.x, size.y);
|
||||||
logD("onMeasure(). New size: " + size.x + "x" + size.y);
|
logD("onMeasure(). New size: " + size.x + "x" + size.y);
|
||||||
}
|
}
|
||||||
@ -237,7 +211,6 @@ public class SurfaceViewRenderer
|
|||||||
|
|
||||||
private void updateSurfaceSize() {
|
private void updateSurfaceSize() {
|
||||||
ThreadUtils.checkIsOnMainThread();
|
ThreadUtils.checkIsOnMainThread();
|
||||||
synchronized (layoutLock) {
|
|
||||||
if (enableFixedSize && rotatedFrameWidth != 0 && rotatedFrameHeight != 0 && getWidth() != 0
|
if (enableFixedSize && rotatedFrameWidth != 0 && rotatedFrameHeight != 0 && getWidth() != 0
|
||||||
&& getHeight() != 0) {
|
&& getHeight() != 0) {
|
||||||
final float layoutAspectRatio = getWidth() / (float) getHeight();
|
final float layoutAspectRatio = getWidth() / (float) getHeight();
|
||||||
@ -267,39 +240,24 @@ public class SurfaceViewRenderer
|
|||||||
getHolder().setSizeFromLayout();
|
getHolder().setSizeFromLayout();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// SurfaceHolder.Callback interface.
|
// SurfaceHolder.Callback interface.
|
||||||
@Override
|
@Override
|
||||||
public void surfaceCreated(final SurfaceHolder holder) {
|
public void surfaceCreated(final SurfaceHolder holder) {
|
||||||
ThreadUtils.checkIsOnMainThread();
|
ThreadUtils.checkIsOnMainThread();
|
||||||
eglRenderer.createEglSurface(holder.getSurface());
|
|
||||||
surfaceWidth = surfaceHeight = 0;
|
surfaceWidth = surfaceHeight = 0;
|
||||||
updateSurfaceSize();
|
updateSurfaceSize();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void surfaceDestroyed(SurfaceHolder holder) {
|
public void surfaceDestroyed(SurfaceHolder holder) {}
|
||||||
ThreadUtils.checkIsOnMainThread();
|
|
||||||
final CountDownLatch completionLatch = new CountDownLatch(1);
|
|
||||||
eglRenderer.releaseEglSurface(new Runnable() {
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
completionLatch.countDown();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
ThreadUtils.awaitUninterruptibly(completionLatch);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
|
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {}
|
||||||
ThreadUtils.checkIsOnMainThread();
|
|
||||||
logD("surfaceChanged: format: " + format + " size: " + width + "x" + height);
|
|
||||||
}
|
|
||||||
|
|
||||||
private String getResourceName() {
|
private String getResourceName() {
|
||||||
try {
|
try {
|
||||||
return getResources().getResourceEntryName(getId()) + ": ";
|
return getResources().getResourceEntryName(getId());
|
||||||
} catch (NotFoundException e) {
|
} catch (NotFoundException e) {
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
@ -312,74 +270,38 @@ public class SurfaceViewRenderer
|
|||||||
eglRenderer.clearImage();
|
eglRenderer.clearImage();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update frame dimensions and report any changes to |rendererEvents|.
|
@Override
|
||||||
private void updateFrameDimensionsAndReportEvents(VideoRenderer.I420Frame frame) {
|
public void onFirstFrameRendered() {
|
||||||
synchronized (layoutLock) {
|
|
||||||
if (isRenderingPaused) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (!isFirstFrameRendered) {
|
|
||||||
isFirstFrameRendered = true;
|
|
||||||
logD("Reporting first rendered frame.");
|
|
||||||
if (rendererEvents != null) {
|
if (rendererEvents != null) {
|
||||||
rendererEvents.onFirstFrameRendered();
|
rendererEvents.onFirstFrameRendered();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (rotatedFrameWidth != frame.rotatedWidth() || rotatedFrameHeight != frame.rotatedHeight()
|
|
||||||
|| frameRotation != frame.rotationDegree) {
|
|
||||||
logD("Reporting frame resolution changed to " + frame.width + "x" + frame.height
|
|
||||||
+ " with rotation " + frame.rotationDegree);
|
|
||||||
if (rendererEvents != null) {
|
|
||||||
rendererEvents.onFrameResolutionChanged(frame.width, frame.height, frame.rotationDegree);
|
|
||||||
}
|
|
||||||
rotatedFrameWidth = frame.rotatedWidth();
|
|
||||||
rotatedFrameHeight = frame.rotatedHeight();
|
|
||||||
frameRotation = frame.rotationDegree;
|
|
||||||
post(new Runnable() {
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
updateSurfaceSize();
|
|
||||||
requestLayout();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update frame dimensions and report any changes to |rendererEvents|.
|
@Override
|
||||||
private void updateFrameDimensionsAndReportEvents(VideoFrame frame) {
|
public void onFrameResolutionChanged(int videoWidth, int videoHeight, int rotation) {
|
||||||
synchronized (layoutLock) {
|
|
||||||
if (isRenderingPaused) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (!isFirstFrameRendered) {
|
|
||||||
isFirstFrameRendered = true;
|
|
||||||
logD("Reporting first rendered frame.");
|
|
||||||
if (rendererEvents != null) {
|
if (rendererEvents != null) {
|
||||||
rendererEvents.onFirstFrameRendered();
|
rendererEvents.onFrameResolutionChanged(videoWidth, videoHeight, rotation);
|
||||||
}
|
}
|
||||||
}
|
int rotatedWidth = rotation == 0 || rotation == 180 ? videoWidth : videoHeight;
|
||||||
if (rotatedFrameWidth != frame.getRotatedWidth()
|
int rotatedHeight = rotation == 0 || rotation == 180 ? videoHeight : videoWidth;
|
||||||
|| rotatedFrameHeight != frame.getRotatedHeight()
|
// run immediately if possible for ui thread tests
|
||||||
|| frameRotation != frame.getRotation()) {
|
postOrRun(() -> {
|
||||||
logD("Reporting frame resolution changed to " + frame.getBuffer().getWidth() + "x"
|
rotatedFrameWidth = rotatedWidth;
|
||||||
+ frame.getBuffer().getHeight() + " with rotation " + frame.getRotation());
|
rotatedFrameHeight = rotatedHeight;
|
||||||
if (rendererEvents != null) {
|
|
||||||
rendererEvents.onFrameResolutionChanged(
|
|
||||||
frame.getBuffer().getWidth(), frame.getBuffer().getHeight(), frame.getRotation());
|
|
||||||
}
|
|
||||||
rotatedFrameWidth = frame.getRotatedWidth();
|
|
||||||
rotatedFrameHeight = frame.getRotatedHeight();
|
|
||||||
frameRotation = frame.getRotation();
|
|
||||||
post(() -> {
|
|
||||||
updateSurfaceSize();
|
updateSurfaceSize();
|
||||||
requestLayout();
|
requestLayout();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void postOrRun(Runnable r) {
|
||||||
|
if (Thread.currentThread() == Looper.getMainLooper().getThread()) {
|
||||||
|
r.run();
|
||||||
|
} else {
|
||||||
|
post(r);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void logD(String string) {
|
private void logD(String string) {
|
||||||
Logging.d(TAG, resourceName + string);
|
Logging.d(TAG, resourceName + ": " + string);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user