如何使用com.android.camera.action.CROP设置输出图像
image
image-processing
6
0

我有裁剪图像的代码,如下所示:

public void doCrop(){
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, "Cant find crop app").show();
   return;
} else{
   intent.setData(selectImageUri);
   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);
   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_RESULT);
   }
}
}

public void onActivityResult (int requestCode, int resultCode, Intent dara){
   if (resultCode == RESULT_OK){
      if (requestCode == CROP_RESULT){
          Bundle extras = data.getExtras();
          if (extras != null){
              bmp = extras.getParcelable("data");
          }
          File f = new File(selectImageUri.getPath());
          if (f.exists()) f.delete();
          Intent inten3 = new Intent(this, tabActivity.class);
          startActivity(inten3);
      }
   }
}

根据我的阅读,代码intent.putExtra("outputX", 300); intent.putExtra("outputY", 300);用于设置裁剪结果的分辨率,但是为什么我无法获得比300x300高的结果图像分辨率?当我设置intent.putExtra("outputX", 800); intent.putExtra("outputY", 800);裁剪功能没有结果或崩溃,这种情况有什么想法吗?

日志猫说“ !!!失败的绑定交易!!!!

参考资料:
Stack Overflow
收藏
评论
共 2 个回答
高赞 时间 活跃

该Intent并非公共Android API的一部分,因此不能保证在所有设备上均可使用。它已在android 1.x和2.x的早期版本中使用,但不再使用,因此不建议使用。这可能就是为什么它在整个地方崩溃或工作不正常的原因。

使用诸如Bitmap.createBitmap(..)Bitmap.createScaledBitmap(..)来创建原始图像的调整大小或裁剪版本。这些是Android API的一部分,可以保证正常工作。

在此处此处查看官方文档

要裁剪位图,可以使用Bitmap.createBitmap(Bitmap, int x, int y, int width, int height) 。例如,如果您需要从位图的每一侧裁剪10个像素,请使用以下命令:

Bitmap croppedBitmap = Bitmap.createBitmap(originalBitmap, 10, 10, originalBitmap.getWidth() - 20, originalBitmap.getHeight() - 20);

如果需要向用户显示选择器。然后,您可以执行以下操作:

private static final String TEMP_PHOTO_FILE = "temporary_holder.jpg";  

Intent photoPickerIntent = new Intent(Intent.ACTION_PICK, android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
photoPickerIntent.setType("image/*");
photoPickerIntent.putExtra("crop", "true");
photoPickerIntent.putExtra(MediaStore.EXTRA_OUTPUT, getTempUri());
photoPickerIntent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
startActivityForResult(photoPickerIntent, REQ_CODE_PICK_IMAGE);


    private Uri getTempUri() {
    return Uri.fromFile(getTempFile());
    }

    private File getTempFile() {
    if (isSDCARDMounted()) {

    File f = new File(Environment.getExternalStorageDirectory(),TEMP_PHOTO_FILE);
    try {
    f.createNewFile();
    } catch (IOException e) {

    }
    return f;
    } else {
    return null;
    }
    }

    private boolean isSDCARDMounted(){
    String status = Environment.getExternalStorageState();
    if (status.equals(Environment.MEDIA_MOUNTED))
    return true;
    return false;
    }




protected void onActivityResult(int requestCode, int resultCode,
        Intent imageReturnedIntent) {
    super.onActivityResult(requestCode, resultCode, imageReturnedIntent);

    switch (requestCode) {
    case REQ_CODE_PICK_IMAGE:
        if (resultCode == RESULT_OK) {  
          if (imageReturnedIntent!=null){



               File tempFile = getTempFile();

              String filePath= Environment.getExternalStorageDirectory()
            + "/temporary_holder.jpg";
              System.out.println("path "+filePath);


    Bitmap selectedImage =  BitmapFactory.decodeFile(filePath);
    _image = (ImageView) findViewById(R.id.image);
    _image.setImageBitmap(selectedImage );

}
}
}

这里的代码

收藏
评论

这个问题遍及整个stackoverflow。我很高兴,因为最近我不得不自己解决这个问题。我会尽我所能标记一些重复的副本,但我更喜欢此副本,因为它解决了图像尺寸有限的问题。

简短答案

简短的答案是不使用return-data选项。请在此处阅读有关该选项以及如何获取图像的更多信息: http : //www.androidworks.com/crop_large_photos_with_android 。本文在列出Intent的(已知)配置选项以及如何使用它们方面做得很好。

选项#2:如果将return-data设置为“ false”,则不会从onActivityResult Intent内联返回位图,而是需要将MediaStore.EXTRA_OUTPUT设置为Uri(仅适用于File方案)想要存储位图。这有一些限制,首先,您需要具有一个临时文件系统位置才能提供文件方案URI,这不是一个大问题(除非在某些没有sdcard的设备上)。

    /** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    thiz = this;
    setContentView(R.layout.main);
    mBtn = (Button) findViewById(R.id.btnLaunch);
    photo = (ImageView) findViewById(R.id.imgPhoto);
    mBtn.setOnClickListener(new OnClickListener(){

        public void onClick(View v) {
            try {
                // Launch picker to choose photo for selected contact
                Intent intent = new Intent(Intent.ACTION_GET_CONTENT, null);
                intent.setType("image/*");
                intent.putExtra("crop", "true");
                intent.putExtra("aspectX", aspectX);
                intent.putExtra("aspectY", aspectY);
                intent.putExtra("outputX", outputX);
                intent.putExtra("outputY", outputY);
                intent.putExtra("scale", scale);
                intent.putExtra("return-data", return_data);
                intent.putExtra(MediaStore.EXTRA_OUTPUT, getTempUri());
                intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
                intent.putExtra("noFaceDetection",!faceDetection); // lol, negative boolean noFaceDetection
                if (circleCrop) {
                    intent.putExtra("circleCrop", true);
                }

                startActivityForResult(intent, PHOTO_PICKED);
            } catch (ActivityNotFoundException e) {
                Toast.makeText(thiz, R.string.photoPickerNotFoundText, Toast.LENGTH_LONG).show();
            }
        }
    });
}

private Uri getTempUri() {
    return Uri.fromFile(getTempFile());
}

private File getTempFile() {
    if (isSDCARDMounted()) {
        File f = new File(Environment.getExternalStorageDirectory(),TEMP_PHOTO_FILE);
        try {
            f.createNewFile();
        } catch (IOException e) {
            // TODO Auto-generated catch block
            Toast.makeText(thiz, R.string.fileIOIssue, Toast.LENGTH_LONG).show();
        }
        return f;
    } else {
        return null;
    }
}

private boolean isSDCARDMounted(){
    String status = Environment.getExternalStorageState();    
    if (status.equals(Environment.MEDIA_MOUNTED))
        return true;
    return false;
}

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);

    switch (requestCode) {
        case PHOTO_PICKED:
            if (resultCode == RESULT_OK) {
                if (data == null) {
                    Log.w(TAG, "Null data, but RESULT_OK, from image picker!");
                    Toast.makeText(this, R.string.no_photo_picked,
Toast.LENGTH_SHORT).show();
                    return;
                }

            final Bundle extras = data.getExtras();
            if (extras != null) {
                File tempFile = getTempFile();
                // new logic to get the photo from a URI
                if (data.getAction() != null) {
                    processPhotoUpdate(tempFile);
                }
            }
        }
        break;
    }
}

来自以下代码示例: http : //www.androidworks.com/crop_large_photos_with_android

更多信息

长的答案是根本不使用该意图。继续阅读以找出原因。

非官方API

问题的核心是非官方的意图。非官方的,不是公共API的一部分。目前,它适用于大多数设备,但大多数还不够。 Google也可以在不通知您的情况下随时更改此意图。打破所有使用它的应用程序。就像日历API曾经是非官方的一样。实际上,这种种植意图已经改变了一次。因此,请避免使用此意图。还有其他选择。随时忽略此建议。

只是为了证明“适用于某些设备的声明”,请点击此链接,并让沮丧的Android开发人员讨论应该被视为android核心功能的一部分(不是): https : //code.google.com/p/android /问题/详细信息?id = 1480

时间示例代码?检查此github项目: https : //github.com/lorensiuswlt/AndroidImageCrop

关于尺寸限制

探索此意图时,我遇到的另一个问题是图像裁剪尺寸限制。使用上面的示例代码和300像素以上的图像大小,可以轻松地重现该图像。基本上,这个原始问题是关于什么的。在最佳情况下,您的应用将崩溃。但是我已经看到挂起的设备更糟,这些设备只能通过卸下电池才能重置。

现在,如果删除“返回数据”选项,则可以再次运行它。关于简短内容的更多信息,请参见我已经引用此链接的简短答案: http : //www.androidworks.com/crop_large_photos_with_android

解决方案

有很多问题。问题需要解决。在Google提出公开API之前,唯一可行的解决方案是提供您自己的种植意图。只需在github上找到像这样的合适的作物库即可: https : //github.com/lvillani/android-cropimage

该项目缺少一些文档,但是由于它是非官方android作物意图的摘录,因此您可以使用顶部列出的示例开始。只要确保不使用return-data选项即可。嗯,看看CropImageIntentBuilder类。那应该使您轻松地创建裁剪意图。不要忘记将此活动添加到清单中,并具有写入外部数据存储的权限。

private void doCrop(File croppedResult){
        CropImageIntentBuilder builder = new CropImageIntentBuilder(600,600, croppedResult);
        // don't forget this, the error handling within the library is just ignoring if you do
        builder.setSourceImage(mImageCaptureUri);
        Intent  intent = builder.getIntent(getApplicationContext());
        // do not use return data for big images
        intent.putExtra("return-data", false);
        // start an activity and then get the result back in onActivtyResult
        startActivityForResult(intent, CROP_FROM_CAMERA);
    }

使用该库还为更多定制打开了大门。众所周知,功能上用于调整大小的核心位图是: 如何在android中裁剪已解析的图像?

就是这样。请享用!

收藏
评论
新手导航
  • 社区规范
  • 提出问题
  • 进行投票
  • 个人资料
  • 优化问题
  • 回答问题

关于我们

常见问题

内容许可

联系我们

@2020 AskGo
京ICP备20001863号