日本搞逼视频_黄色一级片免费在线观看_色99久久_性明星video另类hd_欧美77_综合在线视频

國內最全IT社區平臺 聯系我們 | 收藏本站
阿里云優惠2
您當前位置:首頁 > php開源 > 綜合技術 > 【Android開發經驗】設置用戶頭像并裁剪,僅僅是這么簡單?

【Android開發經驗】設置用戶頭像并裁剪,僅僅是這么簡單?

來源:程序員人生   發布時間:2015-02-07 08:40:33 閱讀次數:6583次

    轉載請注明出處:http://blog.csdn.net/zhaokaiqiang1992

    在做APP的時候,如果有用戶系統功能,那末1般都逃不了這個需求,就是給用戶設置頭像,而設置頭像,又包括從拍照和從相冊選取兩個方式,而且選擇了以后,1般又都會要求對圖象進行裁剪,讓用戶設置頭像。今天這篇文章就是介紹如何完成這個需求的。


    我們首先分析1下需求。關于拍照和從相冊選取,都可以向系統發送特定的Intent,喚起對應的系統程序,然后在onActivityResult里面,獲得我們的數據便可。關于圖象裁剪,有兩種方式,1種是自己處理,比如利用第3方的開源項目,如Cropper(https://github.com/edmodo/cropper),來完成我們的需求,另外1種,我們可以直接利用系統提供的裁剪功能,實現圖象的裁剪。


    在代碼實現之前,我們先理理思路。如果是從相冊獲得的照片,在onActivityResult里面,我們可以獲得到兩種情勢的數據,1種是Bitmap,1種是uri。如果Bitmap對象太大的話,可能就直接把我們的程序弄崩了,所以如果相冊中的圖片都是300px以下的圖片,使用bitmap的方式是允許的,也是安全的,但是這在我們的手機里面也是基本不可能的。所以,我推薦不管大小都直接使用uri方式,由于獲得到uri以后,就相當于拿到了圖片的指針,想干嗎就干嗎~


    對直接在相冊獲得圖片來講,其實不會出現太多的問題,但是如果你想使用拍照的圖片進行處理的話,可能就麻煩1些。使用Intent調用起拍照APP以后,我們在onActivityResult里面也能夠獲得到bitmap或是uri,這取決于我們在intent中設置了甚么標志。但是,如果你直接獲得拍照返回的Bitmap的話,可能其實不是你想得到的結果,有可能返回的不是原圖,而是模糊的縮略圖,這是Android系統為了減少內存使用所做的策略,但是我們拿著這張縮略圖是沒法直接用的,所以,從拍照獲得圖片的時候,我們也應當使用uri的方式。


    好了,不管使用哪一種方式,我們都獲得到了所要處理的圖片的uri,那末以后呢?當前是把這個uri作為數據使用Intent發送到進行裁剪的Activity里面,然后裁剪完成以后,在onActivityResult里面,把裁剪以后的bitmap對象設置給ImageView,然后保存起來,上傳到服務器便可。


    了解了全部流程以后,我們就能夠開始寫代碼了。

    如果想喚起相冊,有兩種方式,1種是Intent.ACTION_PICK,還有1種是Intent.ACTION_GET_CONTENT,代碼以下,這兩種方式都可以獲得到圖片的uri數據

//方式1,直接打開圖庫,只能選擇圖庫的圖片 Intent i = new Intent(Intent.ACTION_PICK, MediaStore.Images.Media.EXTERNAL_CONTENT_URI); //方式2,會先讓用戶選擇接收到該要求的APP,可以從文件系統直接選取圖片 Intent intent = new Intent(Intent.ACTION_GET_CONTENT, MediaStore.Images.Media.EXTERNAL_CONTENT_URI); intent.setType("image/*"); startActivityForResult(intent, PICK_FROM_FILE);

    如果我們想從拍照獲得,那末我們就能夠使用下面的方式,先用1個file對象創建1個uri,然后綁定起來,這樣拍照成功,返回以后,我們就能夠根據uri獲得到照片了,而且在file的路徑保存了拍照的結果,下面是代碼實現

Intent intent = new Intent( MediaStore.ACTION_IMAGE_CAPTURE); imgUri = Uri.fromFile(new File(Environment .getExternalStorageDirectory(), "avatar_" + String.valueOf(System.currentTimeMillis()) + ".png")); intent.putExtra(MediaStore.EXTRA_OUTPUT, imgUri); startActivityForResult(intent, PICK_FROM_CAMERA);

    ok,獲得到uri以后,我們應當怎樣辦呢?當前是再打開裁剪界面啦~

    打開裁剪界面的Intent是Intent intent = new Intent("com.android.camera.action.CROP");

    intent中得這些參數含義請參考下圖

附加選項 數據類型 描寫 crop String 發送裁剪信號 aspectX int X方向上的比例 aspectY int Y方向上的比例 outputX int 裁剪區的寬 outputY int 裁剪區的高 scale boolean 是不是保存比例 return-data boolean 是不是將數據保存在Bitmap中返回 data Parcelable 相應的Bitmap數據 circleCrop String 圓形裁剪區域? MediaStore.EXTRA_OUTPUT ("output") URI 將URI指向相應的file:///...,詳見代碼示例

    比較重要的是MediaStore.EXTRA_OUTPUT和return-data這兩個選項。
    如果你將return-data設置為“true”,你將會取得1個與內部數據關聯的Action,并且bitmap以此方式返回:(Bitmap)extras.getParcelable("data")。注意:如果你終究要獲得的圖片非常大,那末此方法會給你帶來麻煩,所以你要控制outputX和outputY保持在較小的尺寸。 

    如果你將return-data設置為“false”,那末在onActivityResult的Intent數據中你將不會接收到任何Bitmap,相反,你需要將MediaStore.EXTRA_OUTPUT關聯到1個Uri,此Uri是用來寄存Bitmap的。 

    了解了這些以后,我們再看下面的代碼應當就很清楚了。

private void doCrop() { final ArrayList<CropOption> cropOptions = new ArrayList<CropOption>(); Intent intent = new Intent("com.android.camera.action.CROP"); intent.setType("image/*"); List<ResolveInfo> list = getPackageManager().queryIntentActivities( intent, 0); int size = list.size(); if (size == 0) { Toast.makeText(this, "can't find crop app", Toast.LENGTH_SHORT) .show(); return; } else { intent.setData(imgUri); intent.putExtra("outputX", 300); intent.putExtra("outputY", 300); intent.putExtra("aspectX", 1); intent.putExtra("aspectY", 1); intent.putExtra("scale", true); intent.putExtra("return-data", true); // only one if (size == 1) { Intent i = new Intent(intent); ResolveInfo res = list.get(0); i.setComponent(new ComponentName(res.activityInfo.packageName, res.activityInfo.name)); startActivityForResult(i, CROP_FROM_CAMERA); } else { // many crop app for (ResolveInfo res : list) { final CropOption co = new CropOption(); co.title = getPackageManager().getApplicationLabel( res.activityInfo.applicationInfo); co.icon = getPackageManager().getApplicationIcon( res.activityInfo.applicationInfo); co.appIntent = new Intent(intent); co.appIntent .setComponent(new ComponentName( res.activityInfo.packageName, res.activityInfo.name)); cropOptions.add(co); } CropOptionAdapter adapter = new CropOptionAdapter( getApplicationContext(), cropOptions); AlertDialog.Builder builder = new AlertDialog.Builder(this); builder.setTitle("choose a app"); builder.setAdapter(adapter, new DialogInterface.OnClickListener() { public void onClick(DialogInterface dialog, int item) { startActivityForResult( cropOptions.get(item).appIntent, CROP_FROM_CAMERA); } }); builder.setOnCancelListener(new DialogInterface.OnCancelListener() { @Override public void onCancel(DialogInterface dialog) { if (imgUri != null) { getContentResolver().delete(imgUri, null, null); imgUri = null; } } }); AlertDialog alert = builder.create(); alert.show(); } } }

    上面的這些代碼,很大的1部份在處理有多個可以接受裁剪intent要求的APP上面,如果你只想用默許的第1個APP,那末這些邏輯你可以刪除。

    在說完這些東西以后,我們在onActivityResult里面可以這樣處理:

@Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { if (resultCode != RESULT_OK) { return; } switch (requestCode) { case PICK_FROM_CAMERA: doCrop(); break; case PICK_FROM_FILE: imgUri = data.getData(); doCrop(); break; case CROP_FROM_CAMERA: if (null != data) { setCropImg(data); } break; } }

    不管從哪里選,都需要進入裁剪,然后裁剪結束以后,我們調用setCropImg(),把裁剪以后的結果設置給ImageView就能夠了,以下

private void setCropImg(Intent picdata) { Bundle bundle = picdata.getExtras(); if (null != bundle) { Bitmap mBitmap = bundle.getParcelable("data"); mImageView.setImageBitmap(mBitmap); saveBitmap(Environment.getExternalStorageDirectory() + "/crop_" + System.currentTimeMillis() + ".png", mBitmap); } }

    saveBitmap是干嗎的?固然是把裁剪以后的Bitmap存起來了!就像下面這樣

public void saveBitmap(String fileName, Bitmap mBitmap) { File f = new File(fileName); FileOutputStream fOut = null; try { f.createNewFile(); fOut = new FileOutputStream(f); mBitmap.compress(Bitmap.CompressFormat.PNG, 100, fOut); fOut.flush(); } catch (Exception e) { e.printStackTrace(); } finally { try { fOut.close(); Toast.makeText(this, "save success", Toast.LENGTH_SHORT).show(); } catch (IOException e) { e.printStackTrace(); } } }

    好了,現在存起來了,就能夠上傳到服務器,完工~

    記得加權限

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

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

代碼下載:https://github.com/ZhaoKaiQiang/CropImageDemo


參考文章:

如何使用Android MediaStore裁剪大圖片

Android大圖片裁剪終極解決方案(上:原理分析) 

Android大圖片裁剪終極解決方案(中:從相冊截圖) 

Android大圖片裁剪終極解決方案(下:拍照截圖)

生活不易,碼農辛苦
如果您覺得本網站對您的學習有所幫助,可以手機掃描二維碼進行捐贈
程序員人生
------分隔線----------------------------
分享到:
------分隔線----------------------------
關閉
程序員人生
主站蜘蛛池模板: 中文字幕成人网 | 永久av | 九九久久国产 | 免费视频在线观看网站 | 精品久久99| 国产精品不卡视频 | 最新av网站在线观看 | a级在线| 国产精品成人国产乱一区 | 日韩成人性视频 | 另类天堂 | 欧美日韩电影一区二区 | 99热最新网址 | 欧美一区二区在线观看 | 日韩91 | 国产一区二区在线看 | 麻豆av在线播放 | 国产亚洲二区 | 高清久久 | 久久久久亚洲一区二区三区 | 91精品国产乱码久久久久久 | 亚洲欧洲成人av每日更新 | 在线日韩 | 激情视频网址 | 日韩精品不卡 | 中文字幕在线观看免费 | 国产视频福利 | 久久久久久久久久久一区二区 | 亚洲综合99| 日韩三级电影在线观看 | 人人九九精 | 免费人成在线观看网站 | 6080亚洲精品一区二区 | 日韩毛片在线观看 | 中文字幕 av在线 | 九色91在线 | av影片在线 | 五月av在线 | 最近中文字幕大全 | 福利在线一区 | 免费视频99|