【Android】20.3 拍照和視頻錄製

来源:http://www.cnblogs.com/rainmj/archive/2016/03/13/5271167.html
-Advertisement-
Play Games

分類:C#、Android、VS2015; 創建日期:2016-03-13 一、簡介 Android提供的Camera有兩個典型的版本,一個是在Android 5.0以前提供的,稱為Camera;另一個是從Android 5.0開始提供的,稱為Camera2。 這裡僅演示使用系統Camera程式實現...


分類:C#、Android、VS2015;

創建日期:2016-03-13

一、簡介

Android提供的Camera有兩個典型的版本,一個是在Android 5.0以前提供的,稱為Camera;另一個是從Android 5.0開始提供的,稱為Camera2。

這裡僅演示使用系統Camera程式實現拍照的基本用法。

二、Camera基本概念

用Camera實現拍照和攝像功能時,有以下兩種實現辦法:

一是直接利用Intent啟動系統自帶的Camera App打開相機實現拍照和攝像功能,然後通過Android.Provider命名空間下的.MediaStore類獲取拍照或攝像結果保存的文件路徑。這是最簡單的實現辦法,對於一般的需求基本上就能滿足要求了。

二是使用Camera API自定義相機,實現專用的拍照和攝像功能。

1、確定是否需要在AndroidMainfest.xml中聲明使用照相機.

(1)Camera許可權和Camera特性

如果你是調用系統Camera程式,不必聲明該許可權和特性。

如果你是構造自定義的Camera程式,則必須在AndroidMainfest.xml中聲明使用照相機的許可權,例如:

<uses-permission android:name="android.permission.CAMERA" />

你還必須在AndroidMainfest.xml中聲明照相機特性,例如:

<uses-feature android:name="android.hardware.camera" />

如果你的程式可能需要使用照相機,但並不是一定要使用,那麼可以設置android:required屬性,比如:

<uses-feature android:name="android.hardware.camera" android:required="false" />

(2)存儲許可權

如果需要在擴展存儲設備上(如sd卡)存儲你的照片或者拍攝的視頻,那麼必須聲明如下許可權:

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

(3)音頻錄製許可權

如果需要錄製音頻或者視頻,必須在AndroidMainfest.xml文件中設置如下許可權:

<uses-permission android:name="android.permission.RECORD_AUDIO" />

2、使用系統Camera程式

這是一種可以讓你的程式以最少的編碼,然後可以快速使用照相機的方法,此方式通過一個Intent來調用系統的Camera程式,讓系統的Camera應用程式去拍攝照片或者錄製視頻,而且可以返回拍攝或者錄製的結果給你自己的應用程式。

(1)基本用法

(a)構造一個Camera Intent -- 創建一個拍攝照片或者視頻的Intent,可以使用如下兩種方法:

向系統Camera程式請求拍攝圖片。例如:

Intent intent = new Intent(MediaStore.ActionImageCapture);

向系統Camera程式請求錄製視頻。例如:

Intent intent = new Intent(MediaStore.ActionVideoCapture);

(b)開啟Camera intent -- 通過調用startActivityForResult()來執行Camera intent, 在你調用這個方法之後,系統Camera程式就是出現在用戶界面,然後用戶就可以拍攝照片或者視頻了。

(c)接收Intent 結果 -- 在你的應用程式裡面建立一個OnActivityResult()方法來接收來自系統Camera程式的執行結果。當用戶拍攝了照片或者視頻(也許取消了拍攝操作),系統就會調用這個方法。

(2)圖片拍攝Intent

使用系統Camera來拍攝圖片,是一種以最快的速度,最少的代碼,讓你的程式能夠拍照的方法。一個圖片拍攝的intent可以包含如下擴展信息:

(a)MediaStore.ExtraOutput -- 指定一個Uri對象,系統Camera程式會把拍攝的圖片存儲到指定位置。這個設置一般是強烈推薦的。如果你不指定這個Uri路徑,那麼系統Camera程式會把圖片以一個預設的名字存儲在一個預設的位置。我們可以通過返回的intent,然後用Intent.GetData()方法來獲取這個值。

例如:

private File file;

string dirPath;

string filePath;

……

Intent imageIntent = new Intent(MediaStore.ActionImageCapture);

filePath = dirPath + $"/PP{System.DateTime.Now:yyyyMMdd_hhmmss}.jpg";

file = new File(filePath);

imageIntent.PutExtra(MediaStore.ExtraOutput, Uri.FromFile(file));

StartActivityForResult(imageIntent, 100, savedInstanceState);

當StartActivityForResult()方法執行之後,用戶就會看到系統的Camera界面。當用戶完成拍攝照片(或者取消拍攝)之後,用戶界面就會返回你自己的應用程式。那麼你必須攔截OnActivityResult方法,以便接收執行結果和繼續你自己代碼的邏輯。

(3)視頻錄製intent

使用系統Camera來錄製視頻,是一種以最快的速度,最少的代碼,讓你的程式能夠錄製視頻的方法。一個視頻錄製的intent可以包含如下擴展信息:

MediaStore.ExtraOutput -- 指定一個Uri對象,系統Camera程式會把錄製的視頻存儲到指定位置。這個設置一般是強烈推薦的。如果你不指定這個Uri路徑,那麼系統Camera程式會把視頻以一個預設的名字存儲在預設的位置。可以通過返回的intent,然後用Intent.GetData()方法來獲取這個值。

MediaStore.ExtraVideoQuality -- 視頻質量,0(低質量,比較小的文件來存儲視頻), 1(高質量,比較大的文件來存儲視頻)

MediaStore.ExtraDurationLimit -- 設置一個值來限制視頻的錄製時間,單位為秒。

MediaStore.ExtraSizeLimit -- 設置一個值來限制視頻大小,byte為單位。

當StartActivityForResult()方法執行之後,用戶就會看到系統的Camera界面。當用戶完成錄製視頻(或者取消拍攝)之後,用戶界面就會返回你自己的應用程式,接下來你就通過重寫的OnActivityResult()方法接收執行結果。

(4)接收Camera返回的結果

一旦你執行了拍攝圖片或者視頻的intent之後,就可以重寫OnActivityResult()方法接收執行結果了。得到結果後,存儲在指定位置的圖片或者視頻就能被我們的程式使用了。

3、構造自定義的Camera程式

調用系統Camera App實現拍照和攝像功能雖然能夠滿足我們的簡單需求,但是畢竟自由度降低了,而且拍照的界面就是系統提供的樣子。

如果你希望自定義比較漂亮的界面,可以自己定製Camera應用程式制自定義的Camera程式可能比調用系統的Camera需要更多的代碼,但是這能給你的用戶帶來不一樣的用戶體驗。比如Camera 360軟體等,就需要根據SDK提供的Camera API來編寫自己的程式。

註意:Android已經棄用了5.0之前的版本中提供的Android.Hardware.Camera類,而是改為用Camera2實現,具體用法請參看相關資料。

二、模擬器設置

由於手機自帶攝像頭,因此可直接使用相機功能。但是,如果你希望在模擬器中測試你的程式,在模擬器中可利用前置相機(Front Camera)來模擬。

下圖是Android 4.4.2(Api 19)的模擬器參數設置界面:

image

將此界面中的【Front Camera】設置為“Emulated”以後,模擬器就會用一個隨機移動的矩形來模擬相機的拍照和攝像效果。

下圖是Android 6.0(Api 23)的模擬器參數設置界面:

image

三、示例

在模擬器中運行時,打開相機後,單擊下方的圓形按鈕即可拍照,拍照的結果保存到SD卡中指定的文件夾下,所拍的圖片也會立即在ImageView控制項中顯示出來,而且還能單獨通過Android系統自帶的【圖庫】程式查看拍照的結果。

1、運行截圖

下圖左側為初始界面,右側為單擊【拍照】後的拍照界面。

image   image

下圖左側為單擊圓形拍照按鈕後的結果,右側為對號後顯示的文件保存位置。

image   image

下麵左圖為單擊【視頻錄製】後的結果,右圖為錄製完成後的結果。

image   image

2、設計步驟

(1)許可權要求

本例子需要在AndroidManifest.xml中添加下麵的許可權:

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

如果已經有對應的許可權就不用再添加了,如果沒有就雙擊Properties文件夾勾選這個許可權,讓系統自動將其添加到AndroidManifest.xml文件中,或者自己直接將其添加到AndroidManifest.xml文件中。

(2)添加ch2003Main.axml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent">
    <Button
        android:id="@+id/ch2003_btn1"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="拍照" />
  <Button
      android:id="@+id/ch2003_btn2"
      android:layout_width="fill_parent"
      android:layout_height="wrap_content"
      android:text="視頻錄製" />
  <TextView
        android:text=""
        android:textAppearance="?android:attr/textAppearanceSmall"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/ch2003_textView1" />
  <ImageView
        android:src="@android:drawable/ic_menu_gallery"
        android:layout_width="fill_parent"
        android:layout_height="300dp"
        android:id="@+id/ch2003_imageView1"
        android:adjustViewBounds="true" />
  <VideoView
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:id="@+id/ch2003_videoView1" />
</LinearLayout>

(3)添加ch2003BitmapHelpers.cs

該文件會自動縮放圖片以適應屏幕大小,這是為瞭解決記憶體不夠時直接載入大圖片可能會導致應用程式崩潰的情況。

using Android.Widget;
using Android.Graphics.Drawables;
using Android.Graphics;

namespace MyDemos.SrcDemos
{
    public static class ch2003BitmapHelpers
    {
        public static void RecycleBitmap(ImageView imageView)
        {
            if (imageView == null)
            {
                return;
            }

            Drawable toRecycle = imageView.Drawable;
            if (toRecycle != null)
            {
                ((BitmapDrawable)toRecycle).Bitmap.Recycle();
            }
        }

        public static Bitmap LoadAndResizeBitmap(string fileName, int width, int height)
        {
            BitmapFactory.Options options = new BitmapFactory.Options
            {
                InPurgeable = true,
                InJustDecodeBounds = true
            };
            BitmapFactory.DecodeFile(fileName, options);
            int outHeight = options.OutHeight;
            int outWidth = options.OutWidth;
            int inSampleSize = 1;
            if (outHeight > height || outWidth > width)
            {
                if(outWidth > outHeight)
                {
                    inSampleSize = outHeight / height;
                }
                else
                {
                    inSampleSize = outWidth / width;
                }
            }
            options.InSampleSize = inSampleSize;
            options.InJustDecodeBounds = false;
            Bitmap resizedBitmap = BitmapFactory.DecodeFile(fileName, options);
            return resizedBitmap;
        }
    }
}

(4)添加ch2003MainActivity.cs

using System.Collections.Generic;
using Android.App;
using Android.Content;
using Android.OS;
using Android.Runtime;
using Android.Widget;
using Android.Provider;
using Android.Content.PM;
using Android.Graphics;

using Android.Net;
using Java.IO;

namespace MyDemos.SrcDemos
{
    [Activity(Label = "ch2003MainActivity")]
    [IntentFilter(new[] { Intent.ActionMain },
        Categories = new[] { Intent.CategoryDefault, ch.MyDemosCategory })]
    public class ch2003MainActivity : Activity
    {
        private File file;
        string dirPath;
        string filePath;
        private ImageView imageView1;
        private VideoView videoView1;
        private TextView textView1;
        private MediaType mediaType;

        protected override void OnCreate(Bundle savedInstanceState)
        {
            base.OnCreate(savedInstanceState);

            SetContentView(Resource.Layout.ch2003Main);
            textView1 = FindViewById<TextView>(Resource.Id.ch2003_textView1);
            imageView1 = FindViewById<ImageView>(Resource.Id.ch2003_imageView1);
            videoView1 = FindViewById<VideoView>(Resource.Id.ch2003_videoView1);
            Button btn1 = FindViewById<Button>(Resource.Id.ch2003_btn1);
            Button btn2 = FindViewById<Button>(Resource.Id.ch2003_btn2);

            //判斷相機是否可用(是否有一個App正在準備拍照)
            if (IsThereAnAppToTakePictures())
            {
                dirPath = $"{Environment.ExternalStorageDirectory.Path}/{Environment.DirectoryPictures}/MyDemosPictures";
                if (!System.IO.Directory.Exists(dirPath))
                {
                    System.IO.Directory.CreateDirectory(dirPath);
                }
                btn1.Click += delegate
                {
                    //拍照
                    mediaType = MediaType.Image;
                    Intent imageIntent = new Intent(MediaStore.ActionImageCapture);
                    filePath = dirPath + $"/PP{System.DateTime.Now:yyyyMMdd_hhmmss}.jpg";
                    file = new File(filePath);
                    imageIntent.PutExtra(MediaStore.ExtraOutput, Uri.FromFile(file));
                    StartActivityForResult(imageIntent, 100);
                };
                btn2.Click += delegate
                {
                    //視頻錄製
                    mediaType = MediaType.Video;
                    Intent videoIntent = new Intent(MediaStore.ActionVideoCapture);
                    filePath = dirPath + $"/PP{System.DateTime.Now:yyyyMMdd_hhmmss}.mp4";
                    file = new File(filePath);
                    videoIntent.PutExtra(MediaStore.ExtraOutput, Uri.FromFile(file));
                    videoIntent.PutExtra(MediaStore.ExtraVideoQuality, 1);
                    StartActivityForResult(videoIntent, 200);
                };
            }
            else
            {
                //如果相機不可用,禁用按鈕
                btn1.Text = "相機不可用,無法拍照";
                btn2.Text = "相機不可用,無法錄製視頻";
                btn1.Enabled = btn2.Enabled = false;
            }
        }

        /// <summary>
        /// 判斷相機是否可用(是否有一個App正在準備拍照)
        /// </summary>
        /// <returns></returns>
        private bool IsThereAnAppToTakePictures()
        {
            Intent intent = new Intent(MediaStore.ActionImageCapture);
            IList<ResolveInfo> availableActivities = PackageManager.QueryIntentActivities(intent, PackageInfoFlags.MatchDefaultOnly);
            return availableActivities != null && availableActivities.Count > 0;
        }

        protected override void OnActivityResult(int requestCode, [GeneratedEnum] Result resultCode, Intent data)
        {
            base.OnActivityResult(requestCode, resultCode, data);

            // 使其可以在圖庫中可用,即瀏覽圖庫時能看到相機拍攝的照片
            Intent mediaScanIntent = new Intent(Intent.ActionMediaScannerScanFile);
            Uri contentUri = Uri.FromFile(file);
            mediaScanIntent.SetData(contentUri);
            SendBroadcast(mediaScanIntent);

            if(mediaType== MediaType.Image)
            {
                imageView1.Visibility = Android.Views.ViewStates.Visible;
                videoView1.Visibility = Android.Views.ViewStates.Gone;
                //輔助程式(ch2003BitmapHelpers.cs)會自動縮放該圖片以適應屏幕大小。
                int width = Resources.DisplayMetrics.WidthPixels;
                int height = imageView1.Height;
                using (Bitmap bitmap = ch2003BitmapHelpers.LoadAndResizeBitmap(filePath, width, height))
                {
                    ch2003BitmapHelpers.RecycleBitmap(imageView1);
                    imageView1.SetImageBitmap(bitmap);
                    imageView1.RefreshDrawableState();
                }
                textView1.Text = "拍照結果:\n" + filePath;
            }
            else
            {
                imageView1.Visibility = Android.Views.ViewStates.Gone;
                videoView1.Visibility = Android.Views.ViewStates.Visible;
                videoView1.SetVideoURI(Uri.Parse(filePath));
                videoView1.SetMediaController(new MediaController(this));
                videoView1.Start();
                textView1.Text = "視頻錄製結果:\n" + filePath;
            }
        }
    }
}

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 列出每個文件的行數 列出代碼行數總和
  •   運行結果截圖:
  • 單機搭建Android開發環境二,安裝ssh、vim和samba,並配置sudo無密碼執行。
  • 由於Fragment的方便性,現在很多人開始大量使用Fragment。 今天使用時遇到各問題,記錄下來並分享下。   使用Fragment都會用FragmentActivity ,特別是在用到ViewPager時。因為需要FramgmentManager, 而我這次有兩個fragment中都用到了V
  • 工具是:JDK環境配置+SDK+ADT工具 一、Activity  主要作用: 1、用戶與應用程式的介面 2、控制項的容器 二、創建Activity要點:(在src中的目錄下包里) 1、一個Activity就是一個類,要繼承android自身的一個類 2、需要覆寫Oncreate方法,第一個運行的Ac
  • 一、Activity的簡要理解     上篇博文已經知道如何編寫一個簡單的Activity了,可能有很多初學者會疑惑到底什麼是Activity?我們來給出Activity的一個通俗的解釋:Activity就是呈現在我們手機上的各種界面,也就是說,只要在手機上我們能看到的,都是Activity。任何一
  •      圖片拉伸 lxx__lxx__lxx__lxx__lxx__lxx__lxx__lxx__lxx__lxx___ lxx__lxx__lxx__lxx__lxx__lxx__lxx__lxx__lxx__lxx___   1. 如果圖片比較大得話,不要用 [UIImage imageNam
  • 前言:這篇GCD的博文是本人閱讀了很多海內外大神的關於GCD的文章,以及結合之前自己對GCD的粗淺的認識,然後取其精華,去其槽粕,綜合起來的筆記,而且是儘可能的以通熟易懂的並且是正確的理論論述方式呈現給讀者,同時也是講大神博客中有的深澀的理論理解的通熟易懂轉述給讀者,已經是儘可能的讓讀者深入理解和掌
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...