1.關於預覽橫豎差90度的問題
原因分析
經過查證和實驗,可以證實:Android提供的SDK(android.hardware.Camera)裡大概不能正常的使用豎屏(portrait layout)載入照相機,當用豎屏模式載入照相機時會產生以下情況:1. 照相機成像左傾90度(傾斜);2. 照相機成像長寬比例不對(失比)。之所以是“大概”,原因是因為可能可以通過一些比較複雜的手段解決。如果以上成立,那為什麼豎屏不能正常成像也就很顯然了。為什麼會產生這樣的情況,請看下面的研究分析。
照相機在一般情況下是必須用landscape layout(橫屏)的,可以證明,先寫一個照相機(只要能preview就行),如果Manifest的activity裡不加入android:screenOrientation="landscape",即預設了 android:screenOrientation="portrait"(豎屏),照相機preview時就會出現左傾90度的現象,並且失比。原因是這樣的(我推測的),網路攝影機對照物的映射是Android底層固定了的,以landscape方式為正,並且產生大小為320*480的像,如果換成portrait方式了,網路攝影機還是產生320*480的像,然後分別對應的放入到一個480*320的屏內,顯然會失比,然後根據豎、橫屏的規則,就產生了左傾90度的情況(圖例見左)。為了進一步證實我對失比原因的推測,我照相機內載入的SurfaceView調成了320*213,比例大概是(320:213)*1.5=(480:320),所成像結果如願的形成左傾但是沒有失比的狀況,這就證實了我的想法。
綜上可以看出,左傾是因為網路攝影機映射產生的,而失比是由於像素比例映射產生的。
解決方案
暫無好的解決方案,只能強制橫屏,記載代碼中加入
setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);
暫無好的解決方案,只能強制橫屏,記載代碼中加入
android:screenOrientation="landscape"
2.關於拍出來的照片不能正確成像,如綠屏,紅綠相間,重疊等
原因分析
有的手機不支援parameter.setPictureSize(width,height)、parameters.setPreviewSize(width,height)方法,為了相容性建議不設定這兩個方法。
附:完整範例程式碼
main.xml布局檔案
<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical" > <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical" android:layout_weight="2"> <SurfaceView android:id="@+id/surfaceView" android:layout_width="fill_parent" android:layout_height="fill_parent" /> </LinearLayout> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="wrap_content" android:orientation="horizontal" android:layout_weight="1" android:background="@android:color/white" android:layout_gravity="center"> <ImageButton android:layout_width="fill_parent" android:layout_height="80dip" android:id="@+id/btnTakePicture" android:layout_gravity="center" android:textSize="30dip" android:layout_weight="1" android:src="@drawable/btn_take_pic"/> <ImageButton android:layout_width="fill_parent" android:layout_height="80dip" android:id="@+id/btnAutoFocus" android:layout_gravity="center" android:textSize="30dip" android:layout_weight="1" android:src="@drawable/btn_auto_focus"/> </LinearLayout></LinearLayout>
二、MainActivity拍照核心代碼
package cn.itcast.takepicture;import java.io.File;import java.io.FileOutputStream;import android.app.Activity;import android.content.pm.ActivityInfo;import android.graphics.Bitmap;import android.graphics.BitmapFactory;import android.graphics.Matrix;import android.graphics.PixelFormat;import android.graphics.Bitmap.CompressFormat;import android.hardware.Camera;import android.hardware.Camera.PictureCallback;import android.os.Bundle;import android.os.Environment;import android.view.SurfaceHolder;import android.view.SurfaceView;import android.view.View;import android.view.Window;import android.view.WindowManager;import android.view.SurfaceHolder.Callback;import android.widget.ImageButton;public class MainActivity extends Activity {private ImageButton btnTakePicture = null;private ImageButton btnAutoFocus = null;private Camera camera = null;@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);// 設定視窗標題requestWindowFeature(Window.FEATURE_NO_TITLE);// 橫屏setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);// 全螢幕顯示getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,WindowManager.LayoutParams.FLAG_FULLSCREEN);// 當此視窗為使用者可見時,保持裝置常開,並保持亮度不變。getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON);setContentView(R.layout.main);SurfaceView surfaceView = (SurfaceView) this.findViewById(R.id.surfaceView);surfaceView.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);surfaceView.getHolder().setFixedSize(320, 240); // 設定解析度surfaceView.getHolder().addCallback(new SurfaceCallback());btnTakePicture = (ImageButton) findViewById(R.id.btnTakePicture);btnAutoFocus = (ImageButton) findViewById(R.id.btnAutoFocus);btnTakePicture.setOnClickListener(onClickListener);btnAutoFocus.setOnClickListener(onClickListener);}private final View.OnClickListener onClickListener = new View.OnClickListener() {@Overridepublic void onClick(View v) {if (v == btnTakePicture) {if (camera != null)camera.takePicture(null, null, new TakePictureCallback()); // 拍照} else if (v == btnAutoFocus) {if (camera != null)camera.autoFocus(null); // 對焦}}};private final class SurfaceCallback implements Callback {private boolean preview; // 是否正在預覽@Overridepublic void surfaceChanged(SurfaceHolder holder, int format, int width,int height) {}@Overridepublic void surfaceCreated(SurfaceHolder holder) {try {camera = Camera.open();Camera.Parameters parameters = camera.getParameters();parameters.setPreviewFrameRate(5); //每秒5幀parameters.setPictureFormat(PixelFormat.JPEG);//設定照片的輸出格式parameters.set("jpeg-quality", 85);//照片品質camera.setParameters(parameters);camera.setPreviewDisplay(holder);camera.startPreview();preview = true;} catch (Exception e) {e.printStackTrace();}}@Overridepublic void surfaceDestroyed(SurfaceHolder holder) {if (camera != null) {if (preview) {camera.stopPreview();preview = false;}camera.release();camera = null; // 記得釋放}}}private final class TakePictureCallback implements PictureCallback {public void onPictureTaken(byte[] data, Camera camera) {Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length);Matrix matrix=new Matrix();//設定縮放matrix.postScale(0.5f, 0.5f);bitmap=Bitmap.createBitmap(bitmap, 0, 0, bitmap.getWidth(), bitmap.getHeight(), matrix, true);File file = new File(Environment.getExternalStorageDirectory(),System.currentTimeMillis() + ".jpg");try {FileOutputStream outStream = new FileOutputStream(file);bitmap.compress(CompressFormat.JPEG, 100, outStream);outStream.close();camera.startPreview();} catch (Exception e) {e.printStackTrace();}}}}
資訊清單檔
<?xml version="1.0" encoding="utf-8"?><manifest xmlns:android="http://schemas.android.com/apk/res/android" package="cn.itcast.takepicture" android:versionCode="1" android:versionName="1.0" > <application android:icon="@drawable/icon" android:label="@string/app_name" > <activity android:label="@string/app_name" android:name=".MainActivity" > <intent-filter > <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> </application> <uses-sdk android:minSdkVersion="7" /> <uses-permission android:name="android.permission.CAMERA" /> <!-- 在SDCard中建立與刪除檔案許可權 --> <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" /> <!-- 往SDCard寫入資料許可權 --> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /></manifest>