Merge branch 'camera_render' into develop

This commit is contained in:
2013-12-03 15:33:23 -04:30
7 changed files with 119 additions and 131 deletions

View File

@@ -10,4 +10,11 @@
android:paddingTop="@dimen/activity_vertical_margin" android:paddingTop="@dimen/activity_vertical_margin"
tools:context=".CamActivity" > tools:context=".CamActivity" >
</LinearLayout> <FrameLayout
android:id="@+id/previewLayout"
android:layout_width="0dip"
android:layout_height="match_parent"
android:layout_weight="0.47" >
</FrameLayout>
</LinearLayout>

View File

@@ -10,7 +10,7 @@
<string name="start_button">Comenzar streaming de video</string> <string name="start_button">Comenzar streaming de video</string>
<string name="title_activity_cam">CamActivity</string> <string name="title_activity_cam">CamActivity</string>
<string name="ipAddressLabel">Dirección IP de NxtAR</string> <string name="ipAddressLabel">Dirección IP de NxtAR</string>
<string name="badIpToast">La dirección IP no es válida</string> <string name="badIpToast">No se encontró un servidor válido</string>
<string name="emptyIpToast">Rellene el campo de dirección IP</string> <string name="emptyIpToast">Rellene el campo de dirección IP</string>
<string name="wifi_on_title">Esta app requiere WiFi</string> <string name="wifi_on_title">Esta app requiere WiFi</string>
<string name="wifi_on_msg">¿Encender el WiFi?</string> <string name="wifi_on_msg">¿Encender el WiFi?</string>
@@ -18,7 +18,7 @@
<string name="wifi_on_button">Encender</string> <string name="wifi_on_button">Encender</string>
<string name="wifi_on_success">El radio del WiFi está encendido</string> <string name="wifi_on_success">El radio del WiFi está encendido</string>
<string name="wifi_on_fail">Esta app no puede funcionar sin wifi, cerrando.</string> <string name="wifi_on_fail">Esta app no puede funcionar sin wifi, cerrando.</string>
<string name= "bt_no_support">Este dispositivo no soporta Bluetooth</string> <string name="bt_no_support">Este dispositivo no soporta Bluetooth</string>
<string name="bt_on_fail">Esta app no puede funcionar sin Bluetooth, cerrando</string> <string name="bt_on_fail">Esta app no puede funcionar sin Bluetooth, cerrando</string>
<string name="lens_icon">Ícono de lente</string> <string name="lens_icon">Ícono de lente</string>
<string name="robot_pair_button">Conectar con el robot</string> <string name="robot_pair_button">Conectar con el robot</string>

View File

@@ -10,7 +10,7 @@
<string name="get_server_button">Connect with NxtAR</string> <string name="get_server_button">Connect with NxtAR</string>
<string name="start_button">Start video streaming</string> <string name="start_button">Start video streaming</string>
<string name="ipAddressLabel">NxtAR IP Address</string> <string name="ipAddressLabel">NxtAR IP Address</string>
<string name="badIpToast">Invalid IP address</string> <string name="badIpToast">No proper server found</string>
<string name="emptyIpToast">Fill out the IP address field</string> <string name="emptyIpToast">Fill out the IP address field</string>
<string name="wifi_on_title">This app requires WiFi</string> <string name="wifi_on_title">This app requires WiFi</string>
<string name="wifi_on_msg">Turn on the WiFi?</string> <string name="wifi_on_msg">Turn on the WiFi?</string>

View File

@@ -2,6 +2,7 @@ package ve.ucv.ciens.ccg.nxtcam;
import ve.ucv.ciens.ccg.nxtcam.camera.CameraPreview; import ve.ucv.ciens.ccg.nxtcam.camera.CameraPreview;
import ve.ucv.ciens.ccg.nxtcam.network.ImageTransferThread; import ve.ucv.ciens.ccg.nxtcam.network.ImageTransferThread;
import ve.ucv.ciens.ccg.nxtcam.network.LCPThread;
import ve.ucv.ciens.ccg.nxtcam.utils.Logger; import ve.ucv.ciens.ccg.nxtcam.utils.Logger;
import android.app.Activity; import android.app.Activity;
import android.content.Intent; import android.content.Intent;
@@ -12,16 +13,18 @@ import android.support.v4.app.NavUtils;
import android.view.Menu; import android.view.Menu;
import android.view.MenuItem; import android.view.MenuItem;
import android.view.WindowManager; import android.view.WindowManager;
import android.widget.FrameLayout;
import android.widget.Toast; import android.widget.Toast;
public class CamActivity extends Activity{ public class CamActivity extends Activity{
private final String TAG = "NXTCAM_CAM"; private final String TAG = "NXTCAM_CAM";
private final String CLASS_NAME = MainActivity.class.getSimpleName(); private final String CLASS_NAME = CamActivity.class.getSimpleName();
private Camera hwCamera; private Camera hwCamera;
private CameraPreview cPreview; private CameraPreview cPreview;
private CameraSetupTask camSetupTask; private CameraSetupTask camSetupTask;
private ImageTransferThread imThread; private ImageTransferThread imThread;
private LCPThread botThread;
private String serverIp; private String serverIp;
/******************* /*******************
@@ -31,12 +34,11 @@ public class CamActivity extends Activity{
protected void onCreate(Bundle savedInstanceState) { protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState); super.onCreate(savedInstanceState);
getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON); getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON);
setContentView(R.layout.activity_cam);
cPreview = new CameraPreview(this, hwCamera);
setContentView(cPreview);
Intent intent = getIntent(); Intent intent = getIntent();
serverIp = intent.getStringExtra("address"); serverIp = intent.getStringExtra("address");
imThread = new ImageTransferThread(serverIp);
} }
@Override @Override
@@ -69,20 +71,28 @@ public class CamActivity extends Activity{
camSetupTask = new CameraSetupTask(); camSetupTask = new CameraSetupTask();
camSetupTask.execute(); camSetupTask.execute();
imThread = new ImageTransferThread(serverIp); // imThread.start();
imThread.start();
} }
@Override @Override
public void onPause(){ public void onPause(){
super.onPause(); super.onPause();
// TODO: Disconnect and destroy the imThread object. // TODO: pause the imThread and botThread objects.
cPreview.removePreviewCallback(); if(cPreview != null){
cPreview.setCamera(null); cPreview.removePreviewCallback();
releaseCamera(); cPreview.setCamera(null);
releaseCamera();
}
}
@Override
public void onDestroy(){
super.onDestroy();
// TODO: Destroy the network threads.
imThread = null;
} }
/****************** /******************
@@ -90,7 +100,11 @@ public class CamActivity extends Activity{
******************/ ******************/
public void startCameraPreview(){ public void startCameraPreview(){
if(hwCamera != null){ if(hwCamera != null){
Logger.log_d(TAG, CLASS_NAME + ".startCameraPreview() :: Setting camera.");
cPreview = new CameraPreview(this, hwCamera);
cPreview.setCamera(hwCamera); cPreview.setCamera(hwCamera);
((FrameLayout)findViewById(R.id.previewLayout)).addView(cPreview);
Logger.log_d(TAG, CLASS_NAME + ".startCameraPreview() :: Camera and content view set.");
}else{ }else{
Logger.log_wtf(TAG, CLASS_NAME + ".startCameraPreview() :: CAMERA IS NULL!"); Logger.log_wtf(TAG, CLASS_NAME + ".startCameraPreview() :: CAMERA IS NULL!");
System.exit(1); System.exit(1);
@@ -110,6 +124,7 @@ public class CamActivity extends Activity{
@Override @Override
protected Camera doInBackground(Void... params) { protected Camera doInBackground(Void... params) {
Camera cam = null; Camera cam = null;
Logger.log_d(TAG, CLASS_NAME + ".doInBackground() :: Opening the camera.");
try{ try{
cam = Camera.open(0); cam = Camera.open(0);
}catch(Exception e){ }catch(Exception e){
@@ -121,7 +136,7 @@ public class CamActivity extends Activity{
@Override @Override
protected void onPostExecute(Camera result) { protected void onPostExecute(Camera result) {
super.onPostExecute(result); super.onPostExecute(result);
hwCamera = result; hwCamera = result;
if(result != null){ if(result != null){
Logger.log_d(TAG, CLASS_NAME + ".onPostExecute() :: Camera successfully opened"); Logger.log_d(TAG, CLASS_NAME + ".onPostExecute() :: Camera successfully opened");

View File

@@ -334,11 +334,6 @@ public class MainActivity extends Activity implements WifiOnDialogListener, Conn
Logger.log_d(TAG, CLASS_NAME + ".doInBackground() :: Packet payload is\n" + received); Logger.log_d(TAG, CLASS_NAME + ".doInBackground() :: Packet payload is\n" + received);
if(received.compareTo("NxtAR server here!") == 0) if(received.compareTo("NxtAR server here!") == 0)
done = true; done = true;
Socket client1, client2;
client1 = new Socket(packet.getAddress(), ProjectConstants.SERVER_TCP_PORT_1);
client1.close();
client2 = new Socket(packet.getAddress(), ProjectConstants.SERVER_TCP_PORT_2);
client2.close();
} }
result = true; result = true;
}catch(IOException io){ }catch(IOException io){
@@ -367,10 +362,10 @@ public class MainActivity extends Activity implements WifiOnDialogListener, Conn
if(packet != null){ if(packet != null){
showToast(R.string.serv_connected, Toast.LENGTH_SHORT); showToast(R.string.serv_connected, Toast.LENGTH_SHORT);
// startCamActivity(result, packet.getAddress().getHostAddress()); startCamActivity(result, packet.getAddress().getHostAddress());
}else{ }else{
showToast(R.string.serv_fail, Toast.LENGTH_SHORT); showToast(R.string.serv_fail, Toast.LENGTH_SHORT);
// startCamActivity(false, null); startCamActivity(false, null);
} }
} }
} }

View File

@@ -7,10 +7,12 @@ public class CameraImageMonitor{
private final String CLASS_NAME = CameraImageMonitor.class.getSimpleName(); private final String CLASS_NAME = CameraImageMonitor.class.getSimpleName();
private byte[] image; private byte[] image;
private boolean imgChanged; private boolean imgProduced;
private boolean imgConsumed;
private CameraImageMonitor(){ private CameraImageMonitor(){
imgChanged = false; imgProduced = false;
imgConsumed = false;
} }
private static class SingletonHolder{ private static class SingletonHolder{
@@ -22,33 +24,40 @@ public class CameraImageMonitor{
} }
public void setImageData(byte[] image){ public void setImageData(byte[] image){
Logger.log_d(TAG, CLASS_NAME + ".setImageData() :: Copying new image."); if(imgConsumed){
synchronized(image){ Logger.log_d(TAG, CLASS_NAME + ".setImageData() :: Copying new image.");
this.image = new byte[image.length]; synchronized(this.image){
System.arraycopy(image, 0, this.image, 0, image.length); //this.image = new byte[image.length];
imgChanged = true; //System.arraycopy(image, 0, this.image, 0, image.length);
image.notifyAll(); this.image = image;
imgProduced = true;
imgConsumed = false;
this.image.notifyAll();
}
Logger.log_d(TAG, CLASS_NAME + ".setImageData() :: Data copy finished.");
}else{
Logger.log_d(TAG, CLASS_NAME + ".setImageData() :: Old image still valid, ignoring new image.");
} }
Logger.log_d(TAG, CLASS_NAME + ".setImageData() :: Data copy finished.");
} }
public byte[] getImageData(){ public byte[] getImageData(){
byte[] returnImg; byte[] returnImg;
Logger.log_d(TAG, CLASS_NAME + ".getImageData() :: Entry point."); Logger.log_d(TAG, CLASS_NAME + ".getImageData() :: Entry point.");
synchronized(image){ synchronized(image){
while(!imgChanged){ while(!imgProduced){
Logger.log_d(TAG, CLASS_NAME + ".getImageData() :: Waiting for new data."); Logger.log_d(TAG, CLASS_NAME + ".getImageData() :: Waiting for new image.");
try{ image.wait(); }catch(InterruptedException ie){} try{ image.wait(); }catch(InterruptedException ie){ }
} }
Logger.log_d(TAG, CLASS_NAME + ".getImageData() :: Retrieving new data."); Logger.log_d(TAG, CLASS_NAME + ".getImageData() :: Retrieving new image.");
returnImg = image.clone(); returnImg = image;
imgChanged = false; imgProduced = false;
imgConsumed = true;
} }
Logger.log_d(TAG, CLASS_NAME + ".getImageData() :: New data retreived."); Logger.log_d(TAG, CLASS_NAME + ".getImageData() :: New image retrieved.");
return returnImg; return returnImg;
} }
public synchronized boolean hasChanged(){ public synchronized boolean hasChanged(){
return imgChanged; return imgProduced;
} }
} }

View File

@@ -13,85 +13,83 @@ import android.os.Build;
import android.view.Surface; import android.view.Surface;
import android.view.SurfaceHolder; import android.view.SurfaceHolder;
import android.view.SurfaceView; import android.view.SurfaceView;
import android.view.View;
import android.view.ViewGroup;
/** A basic Camera preview class */ /** A basic Camera preview class */
@SuppressLint("ViewConstructor") @SuppressLint("ViewConstructor")
public class CameraPreview extends ViewGroup implements SurfaceHolder.Callback, Camera.PreviewCallback { public class CameraPreview extends SurfaceView implements SurfaceHolder.Callback, Camera.PreviewCallback {
private final String TAG = "SURFVIEW"; private final String TAG = "SURFVIEW";
private final String CLASS_NAME = CameraPreview.class.getSimpleName(); private final String CLASS_NAME = CameraPreview.class.getSimpleName();
private Size mPreviewSize; private CameraImageMonitor imgMonitor;
private List<Size> mSupportedPreviewSizes;
private CameraImageMonitor camMonitor;
private Activity parentActivity; private Activity parentActivity;
private SurfaceView mSurfaceView; private SurfaceHolder holder;
private SurfaceHolder mHolder; private Camera camera;
private Camera mCamera;
@SuppressWarnings("deprecation") @SuppressWarnings("deprecation")
public CameraPreview(Context context, Camera camera){ public CameraPreview(Context context, Camera camera){
super(context); super(context);
parentActivity = (Activity)context; parentActivity = (Activity)context;
mSurfaceView = new SurfaceView(context); // surfaceView = new SurfaceView(context);
mHolder = mSurfaceView.getHolder(); holder = getHolder();
mHolder.addCallback(this); holder.addCallback(this);
if(Build.VERSION.SDK_INT <= Build.VERSION_CODES.HONEYCOMB) if(Build.VERSION.SDK_INT <= Build.VERSION_CODES.HONEYCOMB)
mHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS); holder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
} }
public void setCamera(Camera camera){ public void setCamera(Camera camera){
mCamera = camera; this.camera = camera;
if(mCamera != null){ if(this.camera != null){
camMonitor = CameraImageMonitor.getInstance(); Logger.log_d(TAG, CLASS_NAME + ".setCamera() :: Setting camera.");
mSupportedPreviewSizes = mCamera.getParameters().getSupportedPreviewSizes(); imgMonitor = CameraImageMonitor.getInstance();
requestLayout(); requestLayout();
Logger.log_d(TAG, CLASS_NAME + ".setCamera() :: Camera set.");
} }
} }
public void surfaceCreated(SurfaceHolder holder){ public void surfaceCreated(SurfaceHolder holder){
// The Surface has been created, now tell the camera where to draw the preview. // The Surface has been created, now tell the camera where to draw the preview.
Logger.log_d(TAG, CLASS_NAME + ".surfaceCreated() :: Creating surface view.");
try { try {
if(mCamera != null){ if(camera != null)
mCamera.setPreviewDisplay(holder); camera.setPreviewDisplay(holder);
}
} catch (IOException e) { } catch (IOException e) {
Logger.log_d(TAG, "Error setting camera preview: " + e.getMessage()); Logger.log_e(TAG, CLASS_NAME + ".surfaceCreated() :: Error creating camera: " + e.getMessage());
} }
} }
public void surfaceDestroyed(SurfaceHolder holder){ public void surfaceDestroyed(SurfaceHolder holder){
if(mCamera != null){ if(camera != null)
mCamera.stopPreview(); camera.stopPreview();
}
} }
public void surfaceChanged(SurfaceHolder holder, int format, int w, int h){ public void surfaceChanged(SurfaceHolder tmpHolder, int format, int w, int h){
if(mHolder.getSurface() == null){ int result;
int rotation;
int degrees = 0;
Camera.Parameters camParams;
Logger.log_d(TAG, CLASS_NAME + ".surfaceChanged() :: Method started.");
if(this.holder.getSurface() == null || camera == null){
Logger.log_d(TAG, CLASS_NAME + ".surfaceChanged() :: Holder and/or camera are null.");
return; return;
} }
try{ try{ camera.stopPreview(); }catch (Exception e){ }
mCamera.stopPreview();
}catch (Exception e){ }
requestLayout(); requestLayout();
Camera.Parameters camParams = mCamera.getParameters(); camParams = camera.getParameters();
/*Size optimal = getOptimalPreviewSize(camParams.getSupportedPreviewSizes(), w, h); Size optimal = getOptimalPreviewSize(camParams.getSupportedPreviewSizes(), w, h);
if(ProjectConstants.DEBUG) Logger.log_d(TAG, CLASS_NAME + ".surfaceChanged() :: Preview size set at (" + optimal.width + ", " + optimal.height + ")");
Log.d(TAG, CLASS_NAME + ".surfaceChanged() :: Preview size set at (" + optimal.width + ", " + optimal.height + ")");*/ camParams.setPreviewSize(optimal.width, optimal.height);
camParams.setPreviewSize(mPreviewSize.width, mPreviewSize.height); camera.setParameters(camParams);
mCamera.setParameters(camParams);
android.hardware.Camera.CameraInfo info = new android.hardware.Camera.CameraInfo(); android.hardware.Camera.CameraInfo info = new android.hardware.Camera.CameraInfo();
android.hardware.Camera.getCameraInfo(0, info); android.hardware.Camera.getCameraInfo(0, info);
int rotation = parentActivity.getWindowManager().getDefaultDisplay().getRotation(); rotation = parentActivity.getWindowManager().getDefaultDisplay().getRotation();
int degrees = 0;
switch (rotation) { switch (rotation) {
case Surface.ROTATION_0: degrees = 0; break; case Surface.ROTATION_0: degrees = 0; break;
case Surface.ROTATION_90: degrees = 90; break; case Surface.ROTATION_90: degrees = 90; break;
@@ -99,41 +97,41 @@ public class CameraPreview extends ViewGroup implements SurfaceHolder.Callback,
case Surface.ROTATION_270: degrees = 270; break; case Surface.ROTATION_270: degrees = 270; break;
} }
int result;
if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) { if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {
result = (info.orientation + degrees) % 360; result = (info.orientation + degrees) % 360;
result = (360 - result) % 360; // compensate the mirror result = (360 - result) % 360; // compensate the mirror
} else { // back-facing } else { // back-facing
result = (info.orientation - degrees + 360) % 360; result = (info.orientation - degrees + 360) % 360;
} }
mCamera.setDisplayOrientation(result); camera.setDisplayOrientation(result);
camera.setPreviewCallback(this);
mCamera.setPreviewCallback(this);
try { try {
mCamera.setPreviewDisplay(mHolder); camera.setPreviewDisplay(this.holder);
mCamera.startPreview(); camera.startPreview();
}catch (Exception e){ }catch (Exception e){
Logger.log_d(TAG, CLASS_NAME + ".surfaceChanged() :: Error starting camera preview: " + e.getMessage()); Logger.log_e(TAG, CLASS_NAME + ".surfaceChanged() :: Error starting camera preview: " + e.getMessage());
} }
} }
@Override @Override
public void onPreviewFrame(byte[] data, Camera camera){ public void onPreviewFrame(byte[] data, Camera camera){
Logger.log_d(TAG, CLASS_NAME + ".onPreviewFrame() :: Preview received"); Logger.log_d(TAG, CLASS_NAME + ".onPreviewFrame() :: Preview received");
Logger.log_d(TAG, CLASS_NAME + ".onPreviewFrame() :: Frame has" + (camMonitor.hasChanged() ? "" : " not") + " changed."); Logger.log_d(TAG, CLASS_NAME + ".onPreviewFrame() :: Frame has" + (imgMonitor.hasChanged() ? "" : " not") + " changed.");
if(!camMonitor.hasChanged()) if(!imgMonitor.hasChanged())
camMonitor.setImageData(data); imgMonitor.setImageData(data);
} }
public void removePreviewCallback(){ public void removePreviewCallback(){
mCamera.setPreviewCallback(null); if(camera != null)
camera.setPreviewCallback(null);
} }
private Size getOptimalPreviewSize(List<Size> sizes, int w, int h) { private Size getOptimalPreviewSize(List<Size> sizes, int w, int h) {
final double ASPECT_TOLERANCE = 0.1; final double ASPECT_TOLERANCE = 0.1;
double targetRatio = (double) w / h; double targetRatio = (double) w / h;
Logger.log_d(TAG, CLASS_NAME + ".getOptimalPreviewSize() :: Method started.");
if (sizes == null) return null; if (sizes == null) return null;
Size optimalSize = null; Size optimalSize = null;
@@ -161,45 +159,9 @@ public class CameraPreview extends ViewGroup implements SurfaceHolder.Callback,
} }
} }
} }
Logger.log_d(TAG, CLASS_NAME + ".getOptimalPreviewSize() :: Method ended.");
Logger.log_d(TAG, CLASS_NAME + ".getOptimalPreviewSize() :: Optimal size is: (" + Integer.toString(optimalSize.width) +
", " + Integer.toString(optimalSize.height) + ")");
return optimalSize; return optimalSize;
} }
@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
if (changed && getChildCount() > 0) {
final View child = getChildAt(0);
final int width = r - l;
final int height = b - t;
int previewWidth = width;
int previewHeight = height;
if (mPreviewSize != null) {
previewWidth = mPreviewSize.width;
previewHeight = mPreviewSize.height;
}
// Center the child SurfaceView within the parent.
if (width * previewHeight > height * previewWidth) {
final int scaledChildWidth = previewWidth * height / previewHeight;
child.layout((width - scaledChildWidth) / 2, 0,
(width + scaledChildWidth) / 2, height);
} else {
final int scaledChildHeight = previewHeight * width / previewWidth;
child.layout(0, (height - scaledChildHeight) / 2,
width, (height + scaledChildHeight) / 2);
}
}
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
final int width = resolveSize(getSuggestedMinimumWidth(), widthMeasureSpec);
final int height = resolveSize(getSuggestedMinimumHeight(), heightMeasureSpec);
setMeasuredDimension(width, height);
if (mSupportedPreviewSizes != null) {
mPreviewSize = getOptimalPreviewSize(mSupportedPreviewSizes, width, height);
}
}
} }