android之camera用法實(shí)例詳解
本文實(shí)例講述了android之camera用法。分享給大家供大家參考。具體如下:
1.關(guān)于預(yù)覽橫豎差90度的問題
原因分析
經(jīng)過查證和實(shí)驗(yàn),可以證實(shí):Android提供的SDK(android.hardware.Camera)里大概不能正常的使用豎屏(portrait layout)加載照相機(jī),當(dāng)用豎屏模式加載照相機(jī)時(shí)會(huì)產(chǎn)生以下情況:
①. 照相機(jī)成像左傾90度(傾斜);
②. 照相機(jī)成像長寬比例不對(duì)(失比)。
之所以是“大概”,原因是因?yàn)榭赡芸梢酝ㄟ^一些比較復(fù)雜的手段解決。如果以上成立,那為什么豎屏不能正常成像也就很顯然了。為什么會(huì)產(chǎn)生這樣的情況,請(qǐng)看下面的研究分析。
照相機(jī)在一般情況下是必須用landscape layout(橫屏)的,可以證明,先寫一個(gè)照相機(jī)(只要能preview就行),如果Manifest的activity里不加入android:screenOrientation="landscape",即默認(rèn)了 android:screenOrientation="portrait"(豎屏),照相機(jī)preview時(shí)就會(huì)出現(xiàn)左傾90度的現(xiàn)象,并且失比。原因是這樣的(我推測(cè)的),攝像頭對(duì)照物的映射是Android底層固定了的,以landscape方式為正,并且產(chǎn)生大小為320*480的像,如果換成portrait方式了,攝像頭還是產(chǎn)生320*480的像,然后分別對(duì)應(yīng)的放入到一個(gè)480*320的屏內(nèi),顯然會(huì)失比,然后根據(jù)豎、橫屏的規(guī)則,就產(chǎn)生了左傾90度的情況。為了進(jìn)一步證實(shí)我對(duì)失比原因的推測(cè),我照相機(jī)內(nèi)加載的SurfaceView調(diào)成了320*213,比例大概是(320:213)*1.5=(480:320),所成像結(jié)果如愿的形成左傾但是沒有失比的狀況,這就證實(shí)了我的想法。
綜上可以看出,左傾是因?yàn)閿z像頭映射產(chǎn)生的,而失比是由于像素比例映射產(chǎn)生的。
解決方案
暫無好的解決方案,只能強(qiáng)制橫屏,記載代碼中加入
暫無好的解決方案,只能強(qiáng)制橫屏,記載代碼中加入
2.關(guān)于拍出來的照片不能正確成像,如綠屏,紅綠相間,重疊等
原因分析
有的手機(jī)不支持parameter.setPictureSize(width,height)、parameters.setPreviewSize(width,height)方法,為了兼容性建議不設(shè)置這兩個(gè)方法。
附:完整示例代碼:
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; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); // 設(shè)置窗口標(biāo)題 requestWindowFeature(Window.FEATURE_NO_TITLE); // 橫屏 setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE); // 全屏顯示 getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN); // 當(dāng)此窗口為用戶可見時(shí),保持設(shè)備常開,并保持亮度不變。 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); // 設(shè)置分辨率 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() { @Override public 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); // 對(duì)焦 } } }; private final class SurfaceCallback implements Callback { private boolean preview; // 是否正在預(yù)覽 @Override public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) { } @Override public void surfaceCreated(SurfaceHolder holder) { try { camera = Camera.open(); Camera.Parameters parameters = camera.getParameters(); parameters.setPreviewFrameRate(5); //每秒5幀 parameters.setPictureFormat(PixelFormat.JPEG);//設(shè)置照片的輸出格式 parameters.set("jpeg-quality", 85);//照片質(zhì)量 camera.setParameters(parameters); camera.setPreviewDisplay(holder); camera.startPreview(); preview = true; } catch (Exception e) { e.printStackTrace(); } } @Override public 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(); //設(shè)置縮放 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中創(chuàng)建與刪除文件權(quán)限 --> <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" /> <!-- 往SDCard寫入數(shù)據(jù)權(quán)限 --> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> </manifest>
希望本文所述對(duì)大家的Android程序設(shè)計(jì)有所幫助。
- Android編程中調(diào)用Camera時(shí)預(yù)覽畫面有旋轉(zhuǎn)問題的解決方法
- Android camera實(shí)時(shí)預(yù)覽 實(shí)時(shí)處理,人臉識(shí)別示例
- Android自定義照相機(jī)Camera出現(xiàn)黑屏的解決方法
- Android實(shí)現(xiàn)Camera2預(yù)覽和拍照效果
- Android變形(Transform)之Camera使用介紹
- Android Camera是否支持變焦的判斷方法總結(jié)
- android系統(tǒng)在靜音模式下關(guān)閉camera拍照聲音的方法
- Android中使用Camera類編寫手機(jī)拍照App的實(shí)例教程
- Android Camera變焦編程步驟
- Android?Camera2開啟自動(dòng)曝光功能
相關(guān)文章
Flutter 首頁必用組件NestedScrollView的示例詳解
今天介紹的組件是NestedScrollView,大部分的App首頁都會(huì)用到這個(gè)組件。對(duì)Flutter 首頁必用組件NestedScrollView的相關(guān)知識(shí)感興趣的一起看看吧2020-05-05Android實(shí)現(xiàn)調(diào)用攝像頭
本文給大家分享的是,在安卓APP開發(fā)的過程中,經(jīng)常會(huì)需要調(diào)用手機(jī)自身攝像頭拍照的代碼,十分的簡(jiǎn)單實(shí)用,有需要的小伙伴可以參考下。2015-07-07Android利用Theme自定義Activity間的切換動(dòng)畫
這篇文章主要為大家詳細(xì)介紹了Android利用Theme自定義Activity間的切換動(dòng)畫,感興趣的小伙伴們可以參考一下2016-09-09Android入門之動(dòng)態(tài)BroadCast的使用教程
系統(tǒng)自己在很多時(shí)候都會(huì)發(fā)送廣播,比如電量低或者充足,剛啟動(dòng)完,插入耳機(jī),你有一條新的微信消息,這種都是使用BroadCast機(jī)制去實(shí)現(xiàn)的。BroadCast分為靜態(tài)和動(dòng)態(tài)BroadCast兩種,本文就來聊聊動(dòng)態(tài)BroadCast的使用,需要的可以參考一下2022-12-12Android ViewPager實(shí)現(xiàn)無限循環(huán)效果
這篇文章主要為大家詳細(xì)介紹了Android ViewPager實(shí)現(xiàn)無限循環(huán)效果的相關(guān)資料,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-03-03Android 多線程的實(shí)現(xiàn)方法總結(jié)
這篇文章主要介紹了Android 多線程的實(shí)現(xiàn)方法總結(jié)的相關(guān)資料,這里提供三種方法,幫助大家掌握這部分內(nèi)容,需要的朋友可以參考下2017-08-08Android動(dòng)畫學(xué)習(xí)筆記之補(bǔ)間動(dòng)畫
這篇文章主要為大家詳細(xì)介紹了Android動(dòng)畫學(xué)習(xí)筆記之補(bǔ)間動(dòng)畫,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-12-12