目录

1. 序言

1. 需求

2. 环境

2. 功能实现

1. 导包

2.  更改工具类  

3. 直接使用!!

 4. 运行


1. 序言

1. 需求

        毕设是关于翻译的一个Android app,这就包含了图片的翻译,虽然我使用的是有道智云的翻译API,但是Android端对于图片的预处理还是要自己来实现。前面我也写道过利用Android自带的裁剪功能实现简易的图片裁剪,文章地址,但是对于任意比例这个要求,还是小巫见大巫,没什么实质性的作用。

2. 环境

Android studio2021.1.1,Android 11,语言用的Kotlin。

秉着我可以不用,但是我不能不会的想法,我花了1天的时间看了郭霖大神的《第一行代码》第三版其中的kotlin部分,学会了一些皮毛基础。然后发现Kotlin相对于Java比较简洁,再加上Google官方推荐使用Kotlin所以我就转型使用Kotlin来编写项目。(还有一件事)我在该项目中没有使用已经被弃用的startActivityForResult和OnActivityResult方法,而是使用最新推荐的registerForActivityResult(registerForActivityResult的相关文章),如果你想使用startActivityForResult可以参考这篇文章

2. 功能实现

1. 导包

        在module层的build.gradle添加依赖

implementation 'com.soundcloud.android:android-crop:1.0.1@aar'

        这里使用的是GitHub上大佬的项目包,相关地址

2.  更改工具类  

        因为我是用的registerForActivityResult,所以我改写了一个GitHub大佬的一个工具类。

import android.app.Activity
import android.content.ActivityNotFoundException
import android.content.Context
import android.content.Intent
import android.net.Uri
import android.widget.Toast
import androidx.activity.result.ActivityResultLauncher
import com.soundcloud.android.crop.CropImageActivity

class MyCrop private constructor(source: Uri, destination: Uri) {
    private val cropIntent = Intent()

    fun asSquare(): MyCrop {
        cropIntent.putExtra("aspect_x", 1)
        cropIntent.putExtra("aspect_y", 1)
        return this
    }

    fun start(activity: Activity, crop: ActivityResultLauncher<Intent>) {
        //  activity.startActivityForResult(this.getIntent(activity), requestCode);
        crop.launch(getIntent(activity))
    }

    private fun getIntent(context: Context): Intent {
        cropIntent.setClass(context, CropImageActivity::class.java)
        return cropIntent
    }

    companion object {
        const val RESULT_ERROR = 404
        fun of(source: Uri, destination: Uri): MyCrop {
            return MyCrop(source, destination)
        }

        fun getError(result: Intent): Throwable {
            return result.getSerializableExtra("error") as Throwable
        }

        fun pickImage(activity: Activity, pick: ActivityResultLauncher<String>) {
            try {
                pick.launch("image/*")
            } catch (`var`: ActivityNotFoundException) {
                showImagePickerError(activity)
            }
        }

        private fun showImagePickerError(context: Context) {
            Toast.makeText(context, "No image sources available", Toast.LENGTH_SHORT).show()
        }
    }

    init {
        cropIntent.data = source
        cropIntent.putExtra("output", destination)
    }
}

        可以对比一下添加依赖得到的Crop.class,这里我不做详细说明,就是简单的改了一下startActivityForResult的相关代码。

3. 直接使用!!

 新建项目什么的我就不多说了。直接上使用的代码。自己写一个按钮点击响应函数

MyCrop.pickImage(this, pick)///工具类方法,使用Android自带的获取图片功能

其中pick是我定义的一个ActivityResultLauncher<String> 变量

private lateinit var pick: ActivityResultLauncher<String>

这个变量的初始化一定要在OnCreat函数中。

pick = registerForActivityResult(ActivityResultContracts.GetContent()){
            if (it != null) {
                /**
                 * 获取到的图片Uri就是返回回来的it
                 */
                beginCrop(it)
            }
        }

beginCrop函数:

private fun beginCrop(source: Uri) {
        val uriFile = File(filesDir, "cropped.jpeg") //这是app私有的文件存储路径
        val uriTo = Uri.fromFile(uriFile) //私有的路径可以直接用Uri.fromFile获取相应的Uri
        MyCrop.of(source, uriTo).start(this, crop)
    }

crop是我定义的一个ActivityResultLauncher<Intent> 变量(和pick是差不多的):

private lateinit var crop: ActivityResultLauncher<Intent>

/**
* 初始化
*/
crop = registerForActivityResult(ActivityResultContracts.StartActivityForResult()){
    Glide.with(this).load(filesDir.absolutePath + "/cropped.jpeg").into(binding.image)
}

binding.image是一个ImageView控件(关于Viewbinding可以自己查一下,Java Kotlin都可以用),当然也可以直接findViewById

使用的Glide是一款强大的图片显示开源工具,需要添加相应依赖:

implementation 'com.github.bumptech.glide:glide:4.11.0'
    annotationProcessor 'com.github.bumptech.glide:compiler:4.11.0'

最后就是别忘了在AndroidManifest中添加权限:

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

还需要动态添加读取权限,这里就不做赘叙。可以看我之前的文章

 4. 运行

话不多说直接运行

Android 图片裁剪保存失败 android 图片裁剪 任意形状_android-studio

 点击相册中的图片时,app直接闪退,看报错是

Unable to find explicit activity class {com.example.test/com.soundcloud.android.crop.CropImageActivity}; have you declared this activity in your AndroidManifest.xml?

很明显,它说我没在清单中注册这个activity(com.soundcloud.android.crop.CropImageActivity)这个是android-crop的一个用于裁剪的activity,所以在AndroidManifest中注册一下就可以了

<activity android:name="com.soundcloud.android.crop.CropImageActivity"/>

再来!

Android 图片裁剪保存失败 android 图片裁剪 任意形状_kotlin_02

Android 图片裁剪保存失败 android 图片裁剪 任意形状_Android 图片裁剪保存失败_03

点击完成:

Android 图片裁剪保存失败 android 图片裁剪 任意形状_Android 图片裁剪保存失败_04