Android自定义View实现相机对焦框

前言

在相机APP中,手动对焦时都会出现一个对焦框,告诉用户相机正在对焦。那么这种对焦框是怎么实现的呢?
最近项目中有几个需求,实现手动对焦,自动对焦和对焦框。总体来说不是很复杂,在这片文章中我简单介绍一下。

效果展示

对焦框:

对焦框是用自定义view实现的,实现方法还是和以前一样,继承View类,重载构造方法,初始化画笔,在onTouchEvent里面获取当前点击位置的x y轴坐标,在onDraw方法里面调用canvas的drawCircle方法,把我门刚才记录下来的x y轴坐标,半径,画笔等参数传给drawCircle方法就可以画出对焦框了。

点击事件:

如果在外部直接调用view的setOnclickListener方法是获取不到点击事件的,所以点击事件需要使用interface来实现,在外部实现ClickEvent这个接口就可以获取到view的点击事件了。

interface ClickEvent {
         fun clicked()
    }
 
private var clickEvent: ClickEvent? = null 
    
fun setClickEvent(clickEvent: ClickEvent) {
        this.clickEvent = clickEvent
    }

因为相机app刚打开时,对焦框是不可见的,所以我把默认颜色设置成透明色。当view点击时才会显示出来,这里就修改paint的颜色为白色即可。然后对焦完成后把对焦框隐藏掉,把对焦框颜色设置为透明。

其实还有一种方法是可以使用view的visibility属性,调用invisible或gone方法。但是当你修改view的显示状态时会间接地调用invalidate方法,随后onDraw方法就会被调用。我们在onDraw方法里面传递的x y坐标是全局变量,所以每次调用invisible/visible方法会调用onDraw方法,这里传进去的x y坐标是我们上一次点击位置,所以当你点击屏幕时对焦框的位置是不对的,有可能是显示在上次点击的那个位置上。为了避免这种情况就用了改变颜色的方式。

完整代码

import android.annotation.SuppressLint
import android.content.Context
import android.graphics.Canvas
import android.graphics.Color
import android.graphics.Paint
import android.util.AttributeSet
import android.view.MotionEvent
import android.view.View
 
class CircleView@JvmOverloads constructor(
    context: Context,
    attrs: AttributeSet? = null,
    defStyleAttr: Int = 0
) : View(context, attrs, defStyleAttr){
 
    interface ClickEvent {
         fun clicked()
    }
 
    private var clickEvent: ClickEvent? = null
 
    fun setClickEvent(clickEvent: ClickEvent) {
        this.clickEvent = clickEvent
    }
 
    private val paint: Paint
    private val radius = 80f
    private var currentX = 0.0f
    private var currentY = 0.0f
 
    init {
        paint = Paint()
        paint.color = Color.TRANSPARENT
        paint.style = Paint.Style.STROKE
        paint.strokeWidth = 4f
    }
 
    override fun onDraw(canvas: Canvas?) {
        super.onDraw(canvas)
        canvas?.drawCircle(currentX, currentY, radius, paint)
    }
 
    @SuppressLint("ClickableViewAccessibility")
    override fun onTouchEvent(event: MotionEvent?): Boolean {
 
        when (event?.action) {
            MotionEvent.ACTION_DOWN -> {
                currentX = event.x
                currentY = event.y
                paint.color = Color.WHITE
                clickEvent?.clicked()
                invalidate()
            }
        }
        return true
    }
 
    fun invisible(){
        paint.color = Color.TRANSPARENT
        invalidate()
    }
 
}

xml代码

<com.example.camera.CircleView
        android:id="@+id/circle"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_gravity="center">
 

</com.example.camera.CircleView>

Main Activity

private var myAutoFocusCallback : Camera.AutoFocusCallback? = null
 
    private fun focusListener(){
        myAutoFocusCallback = Camera.AutoFocusCallback {isFocused , _ ->
            if (isFocused) {
                Log.d("test", "对焦成功")
                circle.invisible()
                showToast("对焦成功")
            } else {
                Log.d("test", "对焦失败")
            }
        }
    }
 
    private fun getFocus() {
        Log.d("test", "正在对焦")
        camera?.autoFocus(myAutoFocusCallback)
    }
 
    override fun clicked() {
        getFocus()
    }

这里简单介绍介绍一下对焦功能的实现方法:

核心代码是camera的autoFocus方法。

手动对焦:每次点击surface时可以调用camera的autoFocus方法进行对焦。

如果想要实现自动对焦功能,可以借助计时器,比如每隔200毫秒校准一次,校准完成后关闭计时器等。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android自定义相机、预览区域裁剪

    本文实例为大家分享了Android自定义相机,预览区域裁剪的具体代码,供大家参考,具体内容如下 写法一: 预览区域裁剪,方法调用: //按照比例进行裁剪头像区域 Bitmap   resultBitmap = getScaleImage(resultBitmap,  (int) cuttingAreaView.getX(),   (int) cuttingAreaView.getY(),    cuttingAreaView.getWidth(),    cuttingAreaView.getH

  • Android实现小米相机底部滑动指示器

    近期工作内容需要涉及到相机开发,其中一个功能点就是实现一个相机预览页底部的滑动指示器,现在整理出来供大家讨论参考. 先上一张图看下效果: 主要实现功能有: 1.支持左右滑动,每次滑动一个tab 2.支持tab点击,直接跳到对应tab 3.选中的tab一直处于居中位置 4.支持部分UI自定义(大家可根据需要自己改动) 5.tab点击回调 6.内置Tab接口,放入的内容需要实现Tab接口 7.设置预选中tab public class CameraIndicator extends LinearLa

  • Android 调用设备已有的相机应用详情

    目录 1.拍照 1.1 请求相机功能 1.2 调用相机应用拍照获得缩略图 1.3 调用相机应用拍照获得完整图片 1.4 相机应用返回缩略图,完整图片 2.保存文件的配置 3.录制视频 3.1 调用相机应用录制视频 3.2 相机应用返回视频 如果我们只是需要让用户能够拍摄照片,则可以直接请求已有相机应用拍摄照片并将照片返回给我们 1.拍照 1.1 请求相机功能 在清单文件中添加: <manifest ... > <uses-feature android:name="androi

  • Android自定义相机聚焦和显示框

    本文实例为大家分享了Android自定义相机聚焦和显示框的具体代码,供大家参考,具体内容如下 先看使用效果,白色圆框,放大后缩写并变淡隐藏 下面是代码 public class CameraFocusView extends AppCompatImageView { String TAG = getClass().getName(); //显示的圆宽显示的位置 public float currnetX = 40; public float currentY = 50; Paint paint;

  • Android申请相机权限和读写权限实例

    开发一个相机应用,需要申请三个权限:相机.读文件.写文件. 1.在AndroidManifest.xml中添加 <uses-permission android:name="android.permission.CAMERA"/> <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/> <uses-permission android:name

  • Android 自定义view仿微信相机单击拍照长按录视频按钮

    Android仿微信相机的拍照按钮单击拍照,长按录视频.先上效果图. 项目地址:https://github.com/c786909486/PhotoButton2/tree/v1.0 添加依赖 allprojects { repositories { ... maven { url 'https://jitpack.io' } } } dependencies { compile compile 'com.github.c786909486:PhotoButton2:v1.1' } 长按效果分

  • Android自定义view实现阻尼效果的加载动画

    效果: 需要知识: 1. 二次贝塞尔曲线 2. 动画知识 3. 基础自定义view知识 先来解释下什么叫阻尼运动 阻尼振动是指,由于振动系统受到摩擦和介质阻力或其他能耗而使振幅随时间逐渐衰减的振动,又称减幅振动.衰减振动.[1] 不论是弹簧振子还是单摆由于外界的摩擦和介质阻力总是存在,在振动过程中要不断克服外界阻力做功,消耗能量,振幅就会逐渐减小,经过一段时间,振动就会完全停下来.这种振幅随时间减小的振动称为阻尼振动.因为振幅与振动的能量有关,阻尼振动也就是能量不断减少的振动.阻尼振动是非简谐运

  • Android 自定义View 密码框实例代码

    暴露您view中所有影响可见外观的属性或者行为. •通过XML添加和设置样式 •通过元素的属性来控制其外观和行为,支持和重要事件交流的事件监听器 详细步骤见:Android 自定义View步骤 效果图展示: 支持的样式 可以通过XML定义影响外边和行为的属性如下 边框圆角值,边框颜色,分割线颜色,边框宽度,密码长度,密码大小,密码颜色 <declare-styleable name="PasswordInputView"> <attr name="borde

  • Android自定义View详解

    转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/24252901 很多的Android入门程序猿来说对于Android自定义View,可能都是比较恐惧的,但是这又是高手进阶的必经之路,所有准备在自定义View上面花一些功夫,多写一些文章.先总结下自定义View的步骤: 1.自定义View的属性 2.在View的构造方法中获得我们自定义的属性 [ 3.重写onMesure ] 4.重写onDraw 我把3用[]标出了,所以说3不一

  • Android自定义View中attrs.xml的实例详解

    Android自定义View中attrs.xml的实例详解 我们在自定义View的时候通常需要先完成attrs.xml文件 在values中定义一个attrs.xml 然后添加相关属性 这一篇先详细介绍一下attrs.xml的属性. <?xml version="1.0" encoding="utf-8"?> <resources> //自定义属性名,定义公共属性 <attr name="titleText" for

  • Android自定义View 仿QQ侧滑菜单的实现代码

    先看看QQ的侧滑效果 分析一下 先上原理图(不知道能否表达的清楚 ==) -首先这里使用了 Android 的HorizontalScrollView 水平滑动布局作为容器,当然我们需要继承它自定义一个侧滑视图 - 这个容器里面有一个父布局(一般用LinerLayout,本demo用的是),这个父布局里面有且只有两个子控件(布局),初始状态菜单页的位置在Y轴上存在偏移这样可以就可以形成主页叠在菜单页的上方的视觉效果:然后在滑动的过程程中 逐渐修正偏移,最后菜单页和主页并排排列.原理搞清了实现起来

  • Android自定义View绘制随机生成图片验证码

    本篇文章讲的是Android自定义View之随机生成图片验证码,开发中我们会经常需要随机生成图片验证码,但是这个是其次,主要还是想总结一些自定义View的开发过程以及一些需要注意的地方. 按照惯例先看看效果图: 一.先总结下自定义View的步骤: 1.自定义View的属性 2.在View的构造方法中获得我们自定义的属性 3.重写onMesure 4.重写onDraw 其中onMesure方法不一定要重写,但大部分情况下还是需要重写的 二.View 的几个构造函数 1.public CustomV

  • Android自定义View绘制的方法及过程(二)

    上一篇<Android 自定义View(一) Paint.Rect.Canvas介绍>讲了最基础的如何自定义一个View,以及View用到的一些工具类.下面讲下View绘制的方法及过程 public class MyView extends View { private String TAG = "--------MyView"; private int width, height; public MyView(Context context, AttributeSet a

  • Android自定义View实现等级滑动条的实例

     Android自定义View实现等级滑动条的实例 实现效果图: 思路: 首先绘制直线,然后等分直线绘制点: 绘制点的时候把X值存到集合中. 然后绘制背景图片,以及图片上的数字. 点击事件down的时候,换小图片为大图片.move的时候跟随手指移动. up的时候根据此时的X计算最近的集合中的点,然后自动吸附回去. 1,自定义属性 <?xml version="1.0" encoding="utf-8"?> <resources> <de

  • Android自定义View实现loading动画加载效果

    项目开发中对Loading的处理是比较常见的,安卓系统提供的不太美观,引入第三发又太麻烦,这时候自己定义View来实现这个效果,并且进行封装抽取给项目提供统一的loading样式是最好的解决方式了. 先自定义一个View,继承自LinearLayout,在Layout中,添加布局控件 /** * Created by xiedong on 2017/3/7. */ public class Loading_view extends LinearLayout { private Context m

随机推荐