編輯:關於Android編程
在android開發過程中,拍照或者從相冊中選擇圖片是很常見的功能。下面要說得這個案例比較簡單,用戶點擊按鈕選擇拍照或者打開相冊選擇圖片,然後將選中的圖片顯示在手機上。android6.0後,推出了動態權限管理。以往我們將涉及到的權限全部寫在清單文件中,只要用戶安裝了該程序,程序在運行過程中都會獲得相應權限。android6.0後,對於一些特別敏感的權限,開發者必須在程序中進行聲明。拍照和從相冊選擇圖片都是涉及到用戶隱私的敏感權限,必須在程序中進行聲明。
拍照之前,我們需要創建一個文件夾來存放我們拍好的照片。創建好文件夾後,就可以進入拍照的Activity了,代碼如下:
void takePhoto(){ /** * 最後一個參數是文件夾的名稱,可以隨便起 */ File file=new File(Environment.getExternalStorageDirectory(),"拍照"); if(!file.exists()){ file.mkdir(); } /** * 這裡將時間作為不同照片的名稱 */ output=new File(file,System.currentTimeMillis()+".jpg"); /** * 如果該文件夾已經存在,則刪除它,否則創建一個 */ try { if (output.exists()) { output.delete(); } output.createNewFile(); } catch (Exception e) { e.printStackTrace(); } /** * 隱式打開拍照的Activity,並且傳入CROP_PHOTO常量作為拍照結束後回調的標志 * 將文件轉化為uri */ imageUri = Uri.fromFile(output); Intent intent = new Intent("android.media.action.IMAGE_CAPTURE"); intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri); startActivityForResult(intent, CROP_PHOTO); }
我們通過startActivityForResult(intent, CROP_PHOTO)方法進入拍照程序,拍照結束後,會執行onActivityResult(…)方法,所以我們需要重寫該方法獲得拍照Activity給我們返回的數據進而得到照片的bitmap對象。
public void onActivityResult(int req, int res, Intent data) { switch (req) { /** * 拍照的請求標志 */ case CROP_PHOTO: if (res==RESULT_OK) { try { /** * 該uri就是照片文件夾對應的uri */ Bitmap bit = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri)); imageView.setImageBitmap(bit); } catch (Exception e) { Toast.makeText(this,"程序崩潰",Toast.LENGTH_SHORT).show(); } } else{ Log.i("tag", "失敗"); } break; /** * 從相冊中選取圖片的請求標志 */ case REQUEST_CODE_PICK_IMAGE: if (res == RESULT_OK) { try { /** * 該uri是上一個Activity返回的 */ Uri uri = data.getData(); Bitmap bit = BitmapFactory.decodeStream(getContentResolver().openInputStream(uri)); imageView.setImageBitmap(bit); } catch (Exception e) { e.printStackTrace(); Log.d("tag",e.getMessage()); Toast.makeText(this,"程序崩潰",Toast.LENGTH_SHORT).show(); } } else{ Log.i("liang", "失敗"); } break; default: break; } }
至此,拍照就結束了
和拍照非常相似,唯一不同的是打開該Activity的intent不同,打開相冊的代碼如下:
void choosePhoto(){ /** * 打開選擇圖片的界面 */ Intent intent = new Intent(Intent.ACTION_PICK); intent.setType("image/*");//相片類型 startActivityForResult(intent, REQUEST_CODE_PICK_IMAGE); }
選擇好照片後,會執行onActivityResult(…)方法,在主Activity中需要重寫此方法,並且在此方法中得到相冊Activity傳回的數據進而得到照片的bitmap對象。這一過程和拍照的onActivityResult(…)方法是相同的。
無論是拍照還是從相冊中選擇圖片都涉及到用戶的隱私,所以我們需要聲明權限,所以我們需要在清單文件中加入這句:
在android6.0中,這樣做還是不夠的。我們還需要在代碼中動態聲明權限。啟動拍照或者相冊的Activity時,會彈出一個dialog詢問用戶是否同意授權。
只有用戶同意授權後,才能順利完成接下來的操作。需要說明的是,就算用戶拒絕,我們仍然可以打開相機和相冊的Activity,但是操作結束後,並不會返回什麼值,就是說系統拒絕我們讀取數據。
申請動態權限的簡略步驟如下:
用戶無論是同意還是拒絕授予該權限,Activity都會執行onRequestPermissionsResult(…)方法,我們需要在該方法中判斷用戶是否同意該權限。如果同意,執行相應的方法,如果拒絕,最好向用戶解釋下為什麼需要這個權限。
檢查權限是否被授予:<喎?/kf/ware/vc/" target="_blank" class="keylink">vc3Ryb25nPjwvcD4NCjxwcmUgY2xhc3M9"brush:java;">
//第二個參數是需要申請的權限
if (ContextCompat.checkSelfPermission(this,
Manifest.permission.WRITE_EXTERNAL_STORAGE)
!= PackageManager.PERMISSION_GRANTED)
{
//權限還沒有授予,需要在這裡寫申請權限的代碼
}else {
//權限已經被授予,在這裡直接寫要執行的相應方法即可
}
申請權限: 第二個參數是一個字符串數組,裡面是你需要申請的權限。既然是一個數組,那麼就說明你一次可以申請多個權限。由於拍照和選擇相冊圖片只涉及一個權限,所以上面的字符串數組中就只寫了一個權限。最後一個參數是一個整型常量,用於標志你這次申請的權限,該常量在onRequestPermissionsResult(…)方法中會用到。 判斷用戶是否授予該權限 如果用戶同意,那麼就執行相應的方法,如果拒絕,可以向用戶解釋一下申請該權限的原因。
ActivityCompat.requestPermissions(this,
new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
MY_PERMISSIONS_REQUEST_CALL_PHONE2);
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults)
{
if (requestCode == MY_PERMISSIONS_REQUEST_CALL_PHONE)
{
if (grantResults[0] == PackageManager.PERMISSION_GRANTED)
{
takePhoto();
} else
{
// Permission Denied
Toast.makeText(MainActivity.this, "Permission Denied", Toast.LENGTH_SHORT).show();
}
}
if (requestCode == MY_PERMISSIONS_REQUEST_CALL_PHONE2)
{
if (grantResults[0] == PackageManager.PERMISSION_GRANTED)
{
choosePhoto();
} else
{
// Permission Denied
Toast.makeText(MainActivity.this, "Permission Denied", Toast.LENGTH_SHORT).show();
}
}
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
}
4,代碼
布局文件
MainActivity代碼
public class MainActivity extends AppCompatActivity {
private ImageView imageView;
private static final int CROP_PHOTO = 2;
private static final int REQUEST_CODE_PICK_IMAGE=3;
private static final int MY_PERMISSIONS_REQUEST_CALL_PHONE = 6;
private static final int MY_PERMISSIONS_REQUEST_CALL_PHONE2 = 7;
private File output;
private Uri imageUri;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initView();
}
void initView(){
imageView=(ImageView)findViewById(R.id.image);
}
public void takePhone(View view){
if (ContextCompat.checkSelfPermission(this,
Manifest.permission.WRITE_EXTERNAL_STORAGE)
!= PackageManager.PERMISSION_GRANTED)
{
ActivityCompat.requestPermissions(this,
new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
MY_PERMISSIONS_REQUEST_CALL_PHONE2);
}else {
takePhoto();
}
}
public void choosePhone(View view){
if (ContextCompat.checkSelfPermission(this,
Manifest.permission.WRITE_EXTERNAL_STORAGE)
!= PackageManager.PERMISSION_GRANTED)
{
ActivityCompat.requestPermissions(this,
new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
MY_PERMISSIONS_REQUEST_CALL_PHONE2);
}else {
choosePhoto();
}
}
/**
* 拍照
*/
void takePhoto(){
/**
* 最後一個參數是文件夾的名稱,可以隨便起
*/
File file=new File(Environment.getExternalStorageDirectory(),"拍照");
if(!file.exists()){
file.mkdir();
}
/**
* 這裡將時間作為不同照片的名稱
*/
output=new File(file,System.currentTimeMillis()+".jpg");
/**
* 如果該文件夾已經存在,則刪除它,否則創建一個
*/
try {
if (output.exists()) {
output.delete();
}
output.createNewFile();
} catch (Exception e) {
e.printStackTrace();
}
/**
* 隱式打開拍照的Activity,並且傳入CROP_PHOTO常量作為拍照結束後回調的標志
*/
imageUri = Uri.fromFile(output);
Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
startActivityForResult(intent, CROP_PHOTO);
}
/**
* 從相冊選取圖片
*/
void choosePhoto(){
/**
* 打開選擇圖片的界面
*/
Intent intent = new Intent(Intent.ACTION_PICK);
intent.setType("image/*");//相片類型
startActivityForResult(intent, REQUEST_CODE_PICK_IMAGE);
}
public void onActivityResult(int req, int res, Intent data) {
switch (req) {
/**
* 拍照的請求標志
*/
case CROP_PHOTO:
if (res==RESULT_OK) {
try {
/**
* 該uri就是照片文件夾對應的uri
*/
Bitmap bit = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri));
imageView.setImageBitmap(bit);
} catch (Exception e) {
Toast.makeText(this,"程序崩潰",Toast.LENGTH_SHORT).show();
}
}
else{
Log.i("tag", "失敗");
}
break;
/**
* 從相冊中選取圖片的請求標志
*/
case REQUEST_CODE_PICK_IMAGE:
if (res == RESULT_OK) {
try {
/**
* 該uri是上一個Activity返回的
*/
Uri uri = data.getData();
Bitmap bit = BitmapFactory.decodeStream(getContentResolver().openInputStream(uri));
imageView.setImageBitmap(bit);
} catch (Exception e) {
e.printStackTrace();
Log.d("tag",e.getMessage());
Toast.makeText(this,"程序崩潰",Toast.LENGTH_SHORT).show();
}
}
else{
Log.i("liang", "失敗");
}
break;
default:
break;
}
}
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults)
{
if (requestCode == MY_PERMISSIONS_REQUEST_CALL_PHONE)
{
if (grantResults[0] == PackageManager.PERMISSION_GRANTED)
{
takePhoto();
} else
{
// Permission Denied
Toast.makeText(MainActivity.this, "Permission Denied", Toast.LENGTH_SHORT).show();
}
}
if (requestCode == MY_PERMISSIONS_REQUEST_CALL_PHONE2)
{
if (grantResults[0] == PackageManager.PERMISSION_GRANTED)
{
choosePhoto();
} else
{
// Permission Denied
Toast.makeText(MainActivity.this, "Permission Denied", Toast.LENGTH_SHORT).show();
}
}
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
}
}
1.View 繪制流程ViewGroup也是繼承於View,下面看看繪制過程中依次會調用哪些函數。 說明:measure()和onMeasure()在View.Java源
我們在開發安卓App時難免要與服務器打交道,尤其是對於用戶賬號信息的注冊與登錄更是每個Android開發人員必須掌握的技能,本文將對客戶端的注冊/登錄功能的實現進行分析,
本文主要介紹Android4.4默認Home應用Launcher3的啟動過程和Launcher3的數據加載過程。Launcher的啟動是開機時,ActivityManag
在這篇微信公眾平台開發教程中,我們將介紹如何在網頁中實現獲取收貨地址的功能。收貨地址共享接口 在2016年4月13日 進行過升級,2016年5月20日之後只能使用新接口,