因為項目中需要用到頭像上傳的功能,所以就下個Ddmo先來實現下。 demo我是類似仿微信的,在一個GridView中展示所有的圖片,其中第一個item可以去照相;獲取到圖片後再進行剪切。 圖片的剪切是從網上找的感覺不錯就用,暫時也沒有測試。 獲取圖片可以用:https://github.com/lo ...
因為項目中需要用到頭像上傳的功能,所以就下個Ddmo先來實現下。
demo我是類似仿微信的,在一個GridView中展示所有的圖片,其中第一個item可以去照相;獲取到圖片後再進行剪切。
圖片的剪切是從網上找的感覺不錯就用,暫時也沒有測試。
獲取圖片可以用:https://github.com/lovetuzitong/MultiImageSelector來實現
這裡的圓形圖像是用https://github.com/hdodenhof/CircleImageView來實現的
Demo寫的比較粗糙,效果只是在4.4的手機和7.0的模擬器跑了一遍,所以可能會出現問題的。
下載資源:http://download.csdn.net/download/qq_29774291/9955206
如下是demo的效果圖:
如下是選擇圖片中的代碼
通過LoaderManager來獲取到所有的圖片,然後第一個進行拍照的處理
1 package com.item.demo.photo.activity; 2 3 import android.Manifest; 4 import android.app.LoaderManager; 5 import android.content.ContentResolver; 6 import android.content.Context; 7 import android.content.CursorLoader; 8 import android.content.Intent; 9 import android.content.Loader; 10 import android.content.pm.PackageManager; 11 import android.database.Cursor; 12 import android.net.Uri; 13 import android.os.Build; 14 import android.os.Environment; 15 import android.provider.MediaStore; 16 import android.support.annotation.NonNull; 17 import android.support.v4.content.ContextCompat; 18 import android.support.v4.content.FileProvider; 19 import android.support.v7.app.AppCompatActivity; 20 import android.os.Bundle; 21 import android.text.TextUtils; 22 import android.util.Log; 23 import android.view.View; 24 import android.widget.AdapterView; 25 import android.widget.GridView; 26 import android.widget.ImageView; 27 28 import com.item.demo.photo.BuildConfig; 29 import com.item.demo.photo.R; 30 import com.item.demo.photo.adapter.MyPhotoAdapter; 31 import com.item.demo.photo.uilts.Image; 32 import java.io.File; 33 import java.util.ArrayList; 34 import java.util.List; 35 36 /** 37 * 圖片選擇界面 38 */ 39 public class MyPhotoActivity extends AppCompatActivity { 40 41 private static final int REQUEST_CAPTURE = 100; 42 //private static final int REQUEST_PICK = 101; 43 private static final int REQUEST_CROP_PHOTO = 102; 44 45 public static final int FINSH_RESULT = 104;//截圖後的返回 46 47 private static final int LOADER_ID = 0x0100; 48 private LoadCallBack mLoad = new LoadCallBack(); 49 50 private MyPhotoAdapter mAdapter; 51 private List<Image> images = new ArrayList<>(); 52 //調用照相機返回圖片文件 53 private File tempFile; 54 private static final int MIN_IMAGE_FILE_SIZE = 10 * 1024; // 最小的圖片大小 55 56 @Override 57 protected void onCreate(Bundle savedInstanceState) { 58 super.onCreate(savedInstanceState); 59 setContentView(R.layout.activity_my_photo); 60 61 GridView gv_photo = (GridView)findViewById(R.id.gv_photo); 62 ImageView img_back = (ImageView)findViewById(R.id.iv_back); 63 images.add(new Image()); 64 mAdapter = new MyPhotoAdapter(this,images); 65 gv_photo.setAdapter(mAdapter); 66 gv_photo.setOnItemClickListener(new AdapterView.OnItemClickListener() { 67 @Override 68 public void onItemClick(AdapterView<?> adapterView, View view, int i, long l) { 69 if(i == 0){ 70 //第一個就去照相 71 if(hasPermission(new String[]{Manifest.permission.CAMERA,Manifest.permission.WRITE_EXTERNAL_STORAGE})){ 72 gotoCamera(); 73 }else { 74 requestPermission(0x02,new String[]{Manifest.permission.CAMERA,Manifest.permission.WRITE_EXTERNAL_STORAGE}); 75 } 76 }else { 77 //這裡點擊獲取到圖片地址然後裁剪 78 gotoClipActivity(Uri.parse(images.get(i).getPath())); 79 } 80 } 81 }); 82 img_back.setOnClickListener(new View.OnClickListener() { 83 @Override 84 public void onClick(View view) { 85 finish(); 86 } 87 }); 88 } 89 90 @Override 91 protected void onStart() { 92 super.onStart(); 93 94 if(hasPermission(new String[]{Manifest.permission.READ_EXTERNAL_STORAGE})){ 95 getLoaderManager().initLoader(LOADER_ID,null,mLoad); 96 }else { 97 requestPermission(0x01,new String[]{Manifest.permission.READ_EXTERNAL_STORAGE}); 98 } 99 } 100 101 private class LoadCallBack implements LoaderManager.LoaderCallbacks<Cursor>{ 102 private final String[] IMAGE_PROJECTION = new String[]{ 103 MediaStore.Images.Media._ID,//Id 104 MediaStore.Images.Media.DATA,//圖片路徑 105 MediaStore.Images.Media.DATE_ADDED//圖片的創建時間 106 }; 107 108 @Override 109 public Loader<Cursor> onCreateLoader(int id, Bundle args) { 110 //創建一個Loader 111 if(id == LOADER_ID){ 112 //如果是我們的ID則進行初始化 113 return new CursorLoader(getBaseContext(), 114 MediaStore.Images.Media.EXTERNAL_CONTENT_URI, 115 IMAGE_PROJECTION, 116 null, 117 null, 118 IMAGE_PROJECTION[2] + " DESC"); 119 } 120 return null; 121 } 122 123 @Override 124 public void onLoadFinished(Loader<Cursor> loader, Cursor data) { 125 //當Loader載入完成時 126 List<Image> images = new ArrayList<>(); 127 //判斷是否有數據 128 if(data != null){ 129 int count = data.getCount(); 130 if(count > 0){ 131 data.moveToFirst(); 132 // 得到對應的列的Index坐標 133 int indexId = data.getColumnIndexOrThrow(IMAGE_PROJECTION[0]); 134 int indexPath = data.getColumnIndexOrThrow(IMAGE_PROJECTION[1]); 135 int indexDate = data.getColumnIndexOrThrow(IMAGE_PROJECTION[2]); 136 do { 137 // 迴圈讀取,直到沒有下一條數據 138 int id = data.getInt(indexId); 139 String path = data.getString(indexPath); 140 long dateTime = data.getLong(indexDate); 141 142 File file = new File(path); 143 if (!file.exists() || file.length() < MIN_IMAGE_FILE_SIZE) { 144 // 如果沒有圖片,或者圖片大小太小,則跳過 145 continue; 146 } 147 // 添加一條新的數據 148 Image image = new Image(); 149 image.setId(id); 150 image.setPath(path); 151 image.setDate(dateTime); 152 images.add(image); 153 154 } while (data.moveToNext()); 155 } 156 } 157 updateSource(images); 158 } 159 160 @Override 161 public void onLoaderReset(Loader<Cursor> loader) { 162 updateSource(null); 163 } 164 } 165 166 /** 167 * 通知Adapter數據更改的方法 168 * @param images 新的數據 169 */ 170 private void updateSource(List<Image> images){ 171 this.images.clear(); 172 this.images.add(new Image()); 173 if(images == null || images.size() == 0) return; 174 this.images.addAll(images); 175 mAdapter.notifyDataSetChanged(); 176 } 177 178 /** 179 *許可權的返回 180 */ 181 @Override 182 public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { 183 super.onRequestPermissionsResult(requestCode, permissions, grantResults); 184 switch(requestCode){ 185 case 0x02: 186 if(grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED){ 187 gotoCamera(); 188 } 189 break; 190 case 0x01: 191 if(grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED){ 192 getLoaderManager().initLoader(LOADER_ID,null,mLoad); 193 } 194 } 195 } 196 197 @Override 198 protected void onActivityResult(int requestCode, int resultCode, Intent data) { 199 super.onActivityResult(requestCode, resultCode, data); 200 switch (requestCode){ 201 case REQUEST_CAPTURE://系統相機返回 202 if(resultCode == RESULT_OK){ 203 Log.d("jiejie","--------相機---------" + Uri.fromFile(tempFile).toString()); 204 Log.d("jiejie","--------path----------" + getRealFilePathFromUri(MyPhotoActivity.this,Uri.fromFile(tempFile))); 205 gotoClipActivity(Uri.fromFile(tempFile)); 206 } 207 208 break; 209 case REQUEST_CROP_PHOTO: 210 if(resultCode == RESULT_OK){ 211 if(data != null){ 212 Uri uri = data.getData(); 213 Log.d("jiejie","-------------" + data.getData().getPath()); 214 String cropImagePath = getRealFilePathFromUri(MyPhotoActivity.this,uri); 215 Log.d("jiejie","------crop--------" + cropImagePath); 216 Intent intent = new Intent(); 217 intent.putExtra("image",cropImagePath); 218 setResult(FINSH_RESULT,intent); 219 MyPhotoActivity.this.finish(); 220 } 221 222 } 223 break; 224 } 225 } 226 227 /** 228 * 跳轉到系統照相機 229 */ 230 private void gotoCamera(){ 231 String SDState = Environment.getExternalStorageState(); 232 //判斷SD卡是否存在 233 if(SDState.equals(Environment.MEDIA_MOUNTED)){ 234 tempFile = new File(checkDirPath(Environment.getExternalStorageDirectory().getPath()+ "/image/"), System.currentTimeMillis() + ".jpg"); 235 //隱式的打開調用系統相冊 236 Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); 237 if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.N){ 238 intent.setFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION); 239 //如果是7.0及以上的系統使用FileProvider的方式創建一個Uri 240 Uri contentUri = FileProvider.getUriForFile(MyPhotoActivity.this, BuildConfig.APPLICATION_ID + ".fileProvider", tempFile); 241 intent.putExtra(MediaStore.EXTRA_OUTPUT, contentUri); 242 }else { 243 intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(tempFile)); 244 } 245 startActivityForResult(intent,REQUEST_CAPTURE); 246 } 247 } 248 249 /** 250 * 打開截圖的界面 251 * @param uri 252 */ 253 private void gotoClipActivity(Uri uri){ 254 if(uri == null){ 255 return; 256 } 257 Intent intent = new Intent(this,ClipImageActivity.class); 258 intent.putExtra("type",1); 259 intent.setData(uri); 260 startActivityForResult(intent,REQUEST_CROP_PHOTO); 261 } 262 263 /** 264 * 檢查文件是否存在 265 */ 266 private static String checkDirPath(String dirPath) { 267 if (TextUtils.isEmpty(dirPath)) { 268 return ""; 269 } 270 File dir = new File(dirPath); 271 if (!dir.exists()) { 272 dir.mkdirs(); 273 } 274 return dirPath; 275 } 276 /** 277 * 判斷是否有指定的許可權 278 */ 279 public boolean hasPermission(String... permissions) { 280 281 for (String permisson : permissions) { 282 if (ContextCompat.checkSelfPermission(this, permisson) 283 != PackageManager.PERMISSION_GRANTED) { 284 return false; 285 } 286 } 287 return true; 288 } 289 /** 290 * 申請指定的許可權. 291 */ 292 public void requestPermission(int code, String... permissions) { 293 294 if (Build.VERSION.SDK_INT >= 23) { 295 requestPermissions(permissions, code); 296 } 297 } 298 299 /** 300 * 根據Uri返迴文件絕對路徑 301 * 相容了file:///開頭的 和 content://開頭的情況 302 */ 303 public static String getRealFilePathFromUri(final Context context, final Uri uri) { 304 if (null == uri) return null; 305 final String scheme = uri.getScheme(); 306 String data = null; 307 if (scheme == null) 308 data = uri.getPath(); 309 else if (ContentResolver.SCHEME_FILE.equals(scheme)) { 310 data = uri.getPath(); 311 } else if (ContentResolver.SCHEME_CONTENT.equals(scheme)) { 312 Cursor cursor = context.getContentResolver().query(uri, new String[]{MediaStore.Images.ImageColumns.DATA}, null, null, null); 313 if (null != cursor) { 314 if (cursor.moveToFirst()) { 315 int index = cursor.getColumnIndex(MediaStore.Images.ImageColumns.DATA); 316 if (index > -1) { 317 data = cursor.getString(index); 318 } 319 } 320 cursor.close(); 321 } 322 } 323 return data; 324 } 325 }
其中處理動態的許可權還需要添加7.0的照相處理
在清單文件中加如下配置:
<provider android:name="android.support.v4.content.FileProvider" android:authorities="com.item.demo.photo.fileProvider" android:grantUriPermissions="true" android:exported="false"> <meta-data android:name="android.support.FILE_PROVIDER_PATHS" android:resource="@xml/file_paths" /> </provider>
圖片的剪切
package com.item.demo.photo.activity; import android.content.Intent; import android.graphics.Bitmap; import android.net.Uri; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.util.Log; import android.view.View; import android.widget.ImageView; import android.widget.TextView; import com.item.demo.photo.R; import com.item.demo.photo.view.ClipViewLayout; import java.io.File; import java.io.IOException; import java.io.OutputStream; /** * 圖片剪切 */ public class ClipImageActivity extends AppCompatActivity implements View.OnClickListener { private ClipViewLayout clipViewLayout1; private ClipViewLayout clipViewLayout2; private ImageView back; private TextView tv_ok; //類別 1:圓形 2:方形 private int type; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_clip_image); type = getIntent().getIntExtra("type",1); initView(); } private void initView() { clipViewLayout1 = (ClipViewLayout)findViewById(R.id.clipViewLayout1); clipViewLayout2 = (ClipViewLayout)findViewById(R.id.clipViewLayout2); back = (ImageView)findViewById(R.id.iv_back); tv_ok = (TextView)findViewById(R.id.tv_ok); back.setOnClickListener(this); tv_ok.setOnClickListener(this); } @Override protected void onResume() { super.onResume(); if(type == 1){ clipViewLayout1.setVisibility(View.VISIBLE); clipViewLayout2.setVisibility(View.GONE); //設置圖片資源 clipViewLayout1.setImageSrc(getIntent().getData()); }else { clipViewLayout2.setVisibility(View.VISIBLE); clipViewLayout1.setVisibility(View.GONE); clipViewLayout2.setImageSrc(getIntent().getData()); } } @Override public void onClick(View view) { switch (view.getId()){ case R.id.iv_back: finish(); break; case R.id.tv_ok: generateUriAndReturn(); break; } } /** * 生成Uri並且通過setResult返回給打開的Activity */ private void generateUriAndReturn() { //調用返回剪切圖 Bitmap zoomedCropBitmap; if (type == 1) { zoomedCropBitmap = clipViewLayout1.clip(); } else { zoomedCropBitmap = clipViewLayout2.clip(); } if (zoomedCropBitmap == null) { Log.e("android", "zoomedCropBitmap == null"); return; } Uri mSaveUri = Uri.fromFile(new File(getCacheDir(), "cropped_" + System.currentTimeMillis() + ".jpg")); if (mSaveUri != null) { OutputStream outputStream = null; try { outputStream = getContentResolver().openOutputStream(mSaveUri); if (outputStream != null) { zoomedCropBitmap.compress(Bitmap.CompressFormat.JPEG, 90, outputStream); } } catch (IOException ex) { Log.e("android", "Cannot open file: " + mSaveUri, ex); } finally { if (outputStream != null) { try { outputStream.close(); } catch (IOException e) { e.printStackTrace(); } } } Intent intent = new Intent(); intent.setData(mSaveUri); setResult(RESULT_OK, intent); finish(); } } }