上一篇头像从Bomb获取表中获取。
效果图:
这里有效果是通过style风格设置。
<activity android:name=".SelectphotoActivity"
android:theme="@style/DialogStyleBottom"
>
<style name="DialogStyleBottom" parent="android:Theme.Dialog">
<item name="windowActionBar">false</item>
<item name="windowNoTitle">true</item>
<item name="android:windowAnimationStyle">@style/AnimBottom</item>
<item name="android:windowFrame">@null</item>
<!-- 边框 -->
<item name="android:windowIsFloating">false</item>
<!-- 是否浮现在activity之上 -->
<item name="android:windowIsTranslucent">true</item>
<!-- 半透明 -->
<item name="android:windowNoTitle">true</item>
<!-- 无标题 -->
<item name="android:windowBackground">@android:color/transparent</item>
<!-- 背景透明 -->
<item name="android:backgroundDimEnabled">true</item>
<!-- 模糊 -->
</style>
<style name="AnimBottom" parent="@android:style/Animation">
<item name="android:windowEnterAnimation">@anim/push_bottom_in</item>
<item name="android:windowExitAnimation">@anim/push_bottom_out</item>
</style>
布局
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".activity.SelectPhotoActivity">
<LinearLayout
android:id="@+id/pop_layout_select_pic"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:gravity="center_horizontal"
android:orientation="vertical" >
<TextView
android:id="@+id/text_take_photo"
android:layout_width="fill_parent"
android:layout_height="50.0dip"
android:layout_marginLeft="10.0dip"
android:layout_marginRight="10.0dip"
android:background="@drawable/shape_popup_top"
android:gravity="center"
android:orientation="horizontal"
android:text="@string/photograph"
android:textColor="#ff006cff"
android:textSize="18.0sp"/>
<View
android:layout_width="fill_parent"
android:layout_height="1.0dip"
android:layout_marginLeft="10.0dip"
android:layout_marginRight="10.0dip"
style="@style/horizontal_line_style"/>
<TextView
android:id="@+id/text_pick_photo"
android:layout_width="fill_parent"
android:layout_height="50.0dip"
android:layout_marginLeft="10.0dip"
android:layout_marginRight="10.0dip"
android:background="@drawable/shape_popup_bottom"
android:gravity="center"
android:orientation="horizontal"
android:text="@string/album"
android:textColor="#ff006cff"
android:textSize="18.0sp"/>
<TextView
android:id="@+id/text_cancle"
android:layout_width="fill_parent"
android:layout_height="50.0dip"
android:layout_margin="10.0dip"
android:background="@drawable/shape_popup"
android:gravity="center"
android:orientation="horizontal"
android:text="@string/cancel"
android:textColor="#ff006cff"
android:textSize="18.0sp"
android:textStyle="bold"/>
</LinearLayout>
</RelativeLayout>
之后就是从相册里选择或者拍照之后上传到Bomb上。虽然我写的东西网上有很多,但是到自己手里能运行的很少,慢慢学习与尝试。可以是时代久远……
上传文件失败:errorCode:9015,errorMsg:java.io.FileNotFoundException: /storage/emulated/0/DCIM/Camera/IMG_20190602_063607.jpg: open failed: EACCES (Permission denied)
图片上传这里我我遇见一个问题 说查找不到文件路径,权限受限,我也不知道Android路径啊,很懵,我也看不见Android图片的具体路径那,我裁剪了所以找不到吗,还虚拟机的问题那 ,具体这里就是相册的问题,我先做个小dome,测试从相册进行更改ImageView。
测试成功 代码如下:
button1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Intent intent = new Intent(Intent.ACTION_PICK, null);
intent.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, "image/*");
startActivityForResult(intent, CHOOSE_PICTURE);
}
});
button2.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
File outputImage=new File(Environment.getExternalStorageDirectory(),
"output_image.jpg");//创建File对象,用于存储拍照后的图片,获取sd卡根目录
try{
if(outputImage.exists()){
outputImage.delete();
}
outputImage.createNewFile();
}catch (IOException e){
e.printStackTrace();
}
imageUri=Uri.fromFile(outputImage);//File对象转化为Uri对象
Intent intent=new Intent("android.media.action.IMAGE_CAPTURE");
intent.putExtra(MediaStore.EXTRA_OUTPUT,imageUri);
startActivityForResult(intent,TAKE_PICTURE); //启动相机程序
}
});
拍照:创建一个file文件,用来存储摄像头拍下的照片,存放在SD卡根目录下。 同时这里使用的是一个隐式的Intent,调用的startActivityForResult(intent,TAKE_PHOTO); //启动相机程序 来启动活动。
拍下的照片输出成.jpg文件。
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (resultCode == MainActivity.RESULT_OK) {
switch (requestCode) {
case TAKE_PICTURE:
cropPhoto(imageUri);
break;
case CHOOSE_PICTURE:
Uri uri1 = data.getData();
cropPhoto(uri1);
break;
case CROP_SMALL_PICTURE:
Bundle bundle = data.getExtras();
if (bundle != null) {
//在这里获得了剪裁后的Bitmap对象,可以用于上传
Bitmap image = bundle.getParcelable("data");
Log.e("我是裁剪后",""+image);
//设置到ImageView上
img.setImageBitmap(image);
//也可以进行一些保存、压缩等操作后上传
String path = saveImage("crop", image);
Log.e("我是裁剪后2",""+path);
}
break;
}
}
}
如果拍照成功,再次构建一个intent 对象,这个intent 进行裁剪活动 。
public String saveImage(String name, Bitmap bmp) {
File appDir = new File(Environment.getExternalStorageDirectory().getPath());
if (!appDir.exists()) {
appDir.mkdir();
}
String fileName = name + ".jpg";
File file = new File(appDir, fileName);
try {
FileOutputStream fos = new FileOutputStream(file);
bmp.compress(Bitmap.CompressFormat.PNG, 100, fos);
fos.flush();
fos.close();
return file.getAbsolutePath();
} catch (IOException e) {
e.printStackTrace();
}
return null;
}
private Bitmap getURLimage(String imageUrl) {
Bitmap bmp = null;
try {
URL myurl = new URL(imageUrl);
// 获得连接
HttpURLConnection conn = (HttpURLConnection) myurl.openConnection();
conn.setConnectTimeout(6000);//设置超时
conn.setDoInput(true);
conn.setUseCaches(false);//不缓存
conn.connect();
InputStream is = conn.getInputStream();//获得图片的数据流
bmp = BitmapFactory.decodeStream(is);
is.close();
} catch (Exception e) {
e.printStackTrace();
}
return bmp;
}
/**
* 裁剪图片
*/
private void cropPhoto(Uri uri) {
Intent intent = new Intent("com.android.camera.action.CROP");
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
intent.setDataAndType(uri, "image/*");
intent.putExtra("crop", "true");
intent.putExtra("aspectX", 1);
intent.putExtra("aspectY", 1);
intent.putExtra("outputX", 300);
intent.putExtra("outputY", 300);
intent.putExtra("return-data", true);
startActivityForResult(intent, CROP_SMALL_PICTURE);
}
}
这里是将将照片解析成bitmip对象,然后设置到imageview中显示出来。
涉及向SD卡中写数据的操作,声明权限:
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
这里不要写多了,有时候权限写多了也报错,会重复请求。这里我之前多写了获取相机的权限,结果一直报错。
做项目集成:
项目集成就是涉及图片的上传,这里原先有一个找不到路径的我问题,细化之后就解决了问题。
试了很多次才成功的,整套代码:
在一个主界面(主Activity)通过意图跳转至多个不同子Activity上去,当子模块的代码执行完毕后再次返回主页面,将子activity中得到的数据显示在主界面/完成的数据交给主Activity处理。这种带数据的意图跳转需要使用activity的onActivityResult()方法。
(1)startActivityForResult(Intent intent, int requestCode);
第一个参数:一个Intent对象,用于携带将跳转至下一个界面中使用的数据,使用putExtra(A,B)方法,此处存储的数据类型特别多,基本类型全部支持。
第二个参数:如果> = 0,当Activity结束时requestCode将归还在onActivityResult()中。以便确定返回的数据是从哪个Activity中返回,用来标识目标activity。
与下面的resultCode功能一致,感觉Android就是为了保证数据的严格一致性特地设置了两把锁,来保证数据的发送,目的地的严格一致性。
(2)onActivityResult(int requestCode, int resultCode, Intent data)
第一个参数:这个整数requestCode用于与startActivityForResult中的requestCode中值进行比较判断,是以便确认返回的数据是从哪个Activity返回的。
第二个参数:这整数resultCode是由子Activity通过其setResult()方法返回。适用于多个activity都返回数据时,来标识到底是哪一个activity返回的值。
第三个参数:一个Intent对象,带有返回的数据。可以通过data.getXxxExtra( );方法来获取指定数据类型的数据。
二改:
改成点击圆形头像切换头像。SelectPhotoActivity.java
public class SelectPhotoActivity extends BaseActivity {
private Uri imageUri;
private static final int RESULT_CODE_STARTCAMERA = 0x01;
public static final int TAKE_PHOTO = 0x02;
public static final int CHOOSE_PHOTO = 0x03;
public static final int CUT_PHOTO = 0x04;
@Override
protected int contentViewID() {
return R.layout.activity_select_photo;
}
@Override
protected void initialize() {
setTopTitle("", false);
StatusBarUtils.setTransparent(this);
}
@OnClick({R.id.text_take_photo, R.id.text_pick_photo, R.id.text_cancle})
public void OnClick(View view) {
switch (view.getId()) {
case R.id.text_pick_photo:
onAlbum();
break;
case R.id.text_take_photo:
onCamera();
break;
case R.id.text_cancle:
finish();
break;
}
}
private void onAlbum() {
Intent intent = new Intent(Intent.ACTION_PICK, null);
intent.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, "image/*");
startActivityForResult(intent, CHOOSE_PHOTO);
}
private void onCamera() {
if (PackageManager.PERMISSION_GRANTED == ContextCompat.checkSelfPermission(this, Manifest.permission.CAMERA)) {
pickImageFromCamera();
} else {
//提示用户开户权限 拍照和读写sd卡权限
String[] perms = {Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.WRITE_EXTERNAL_STORAGE, Manifest.permission.CAMERA};
ActivityCompat.requestPermissions(this, perms, RESULT_CODE_STARTCAMERA);
}
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
switch (requestCode) {
case RESULT_CODE_STARTCAMERA:
if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
pickImageFromCamera();
} else {
ToastUtils.showShort(SelectPhotoActivity.this, getString(R.string.no_permissions));
}
break;
}
}
//拍照
private void pickImageFromCamera() {
File outputImage = new File(Environment.getExternalStorageDirectory(),
"output_image.jpg");//创建File对象,用于存储拍照后的图片,获取sd卡根目录
try {
if (outputImage.exists()) {
outputImage.delete();
}
outputImage.createNewFile();
} catch (IOException e) {
e.printStackTrace();
}
imageUri = Uri.fromFile(outputImage);//File对象转化为Uri对象
Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
startActivityForResult(intent, TAKE_PHOTO); //启动相机程序
}
@Override
protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (resultCode == RESULT_OK) {
switch (requestCode) {
case TAKE_PHOTO:
cropPhoto(imageUri);
break;
case CUT_PHOTO:
Bundle bundle = data.getExtras();
if (bundle != null) {
//在这里获得了剪裁后的Bitmap对象,可以用于上传
Bitmap image = bundle.getParcelable("data");
String path = saveImage("HeadImages", image);
setPicToView(path);
}
break;
case CHOOSE_PHOTO:
if (data == null || data.getData() == null) {
return;
}
try {
Uri uri = data.getData();
cropPhoto(uri);
} catch (Exception e) {
e.printStackTrace();
}
break;
}
}
}
private void setPicToView(String path) {
if (path == null) {
ToastUtils.showShort(SelectPhotoActivity.this, getString(R.string.image_upload_fail));
return;
}
final BmobFile bmobFile = new BmobFile(new File(path));
bmobFile.uploadblock(new UploadFileListener() {
@Override
public void done(BmobException e) {
if (e == null) {
MyUser myUser = MyUser.getCurrentUser(MyUser.class);
myUser.setPhotoImage(bmobFile);
myUser.update(myUser.getObjectId(), new UpdateListener() {
@Override
public void done(BmobException e) {
if (e == null) {
ToastUtils.showShort(SelectPhotoActivity.this, getString(R.string.image_upload));
EventBus.getDefault().post(new MessageEvent(ConstantConfig.HEAD_IMAGE_SUCCESS));
} else {
ToastUtils.showShort(SelectPhotoActivity.this, getString(R.string.image_upload_fail));
}
}
});
}
}
});
}
/**
* 裁剪图片
*
* @param uri
*/
private void cropPhoto(Uri uri) {
Intent intent = new Intent("com.android.camera.action.CROP");
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
intent.setDataAndType(uri, "image/*");
intent.putExtra("crop", "true");
intent.putExtra("aspectX", 1);
intent.putExtra("aspectY", 1);
intent.putExtra("outputX", 300);
intent.putExtra("outputY", 300);
intent.putExtra("return-data", true);
startActivityForResult(intent, CUT_PHOTO);
}
public String saveImage(String name, Bitmap bmp) {
File appDir = new File(Environment.getExternalStorageDirectory().getPath());
if (!appDir.exists()) {
appDir.mkdir();
}
String fileName = name + ".jpg";
File file = new File(appDir, fileName);
try {
FileOutputStream fos = new FileOutputStream(file);
bmp.compress(Bitmap.CompressFormat.PNG, 100, fos);
fos.flush();
fos.close();
return file.getAbsolutePath();
} catch (IOException e) {
e.printStackTrace();
}
return null;
}
}