首页 > 其他分享 >Canvas->Bitmap绘制

Canvas->Bitmap绘制

时间:2025-01-09 19:00:46浏览次数:3  
标签:Canvas targetRect val canvas bitmap Bitmap 绘制

Canvas.drawBitmap 常用重载方法的详解

/**
 * 如果绘制的Bitmap不需要拉伸,缩放,使用这个方法,比如绘制马赛克底图
 *
 * @param bitmap 要绘制的位图
 * @param left 被绘制的位图左侧的位置
 * @param top 被绘制的位图顶部的位置
 * @param paint 用于绘制位图的 Paint(可以为 null)
 */
public void drawBitmap(@NonNull Bitmap bitmap, float left, float top, @Nullable Paint paint) {
    super.drawBitmap(bitmap, left, top, paint);
}

/**
 * 从Bitmap中拿出一块Rect区域,绘制到另外一块Rect区域上,比如绘制到一个指定的View区域上
 *
 * @param bitmap 要绘制的位图
 * @param src 源矩形,定义位图中要绘制的区域。可以为 null,表示绘制整个位图。
 * @param dst 目标矩形,定义位图在画布上绘制的位置和大小。
 * @param paint 用于绘制位图的 Paint(可以为 null)
 */
public void drawBitmap(@NonNull Bitmap bitmap, @Nullable Rect src, @NonNull Rect dst, @Nullable Paint paint) {
    super.drawBitmap(bitmap, src, dst, paint);
}

/**
 * 通过矩阵变换Bitmap,适合需要复杂变换的场景,比如绘制一张平移、旋转、缩放的Bitmap
 *
 * @param bitmap 要绘制的位图
 * @param matrix 用于变换位图的 Matrix,包括平移、旋转、缩放等操作
 * @param paint 用于绘制位图的 Paint(可以为 null)
 */
public void drawBitmap(@NonNull Bitmap bitmap, @NonNull Matrix matrix, @Nullable Paint paint) {
    super.drawBitmap(bitmap, matrix, paint);
}

如何把Bitmap从原始区域Rect绘制到目标区域Rect上?

  • 一张Bitmap的宽高往往是比手机屏幕宽高大得多,原始Bitmap自身携带一个原始区域Rect,也就是Rect(0, 0, bitmap.width, bitmap.height),如果要绘制到手机屏幕区域某个View的区域,也就是目标Rect,需要使用到Canvas和变换矩阵Matrix的组合使用

方法一

override fun onDraw(canvas: Canvas) {
    // 如果需要在一块指定Rect上绘制平移、旋转、缩放的Bitmap
    bitmap?.let { bitmap ->
        // 计算平移
        val translateX = targetRect.left + (targetRect.width() - bitmap.width) / 2
        val translateY = targetRect.top + (targetRect.height() - bitmap.height) / 2
        // 计算缩放
        val scaleX = targetRect.width() / bitmap.width
        val scaleY = targetRect.height() / bitmap.height
        val scale = min(scaleX, scaleY)
        // 计算旋转
        val degree = 15f
        // 图片原始区域到目标区域的变换矩阵,存储平移、缩放、旋转数值
        val targetMatrix = Matrix().apply {
            postTranslate(translateX, translateY)
            postScale(scale, scale, targetRect.centerX(), targetRect.centerY())
            postRotate(degree, targetRect.centerX(), targetRect.centerY())
            postConcat(手势矩阵)
        }
        
        canvas.concat(targetMatrix) // canvas拼接变换矩阵
        canvas.drawBitmap(bitmap, null, targetRect, Paint())
    }
}

方法二

override fun onDraw(canvas: Canvas) {
    // 如果需要在一块指定Rect上绘制平移、旋转、缩放的Bitmap
    bitmap?.let { bitmap ->
        // 计算平移
        val translateX = targetRect.left + (targetRect.width() - bitmap.width) / 2
        val translateY = targetRect.top + (targetRect.height() - bitmap.height) / 2
        // 计算缩放
        val scaleX = targetRect.width() / bitmap.width
        val scaleY = targetRect.height() / bitmap.height
        val scale = min(scaleX, scaleY)
        // 计算旋转
        val degree = 15f
        // 图片原始区域到目标区域的变换矩阵,存储平移、缩放、旋转数值
        val targetMatrix = Matrix().apply {
            postTranslate(translateX, translateY)
            postScale(scale, scale, targetRect.centerX(), targetRect.centerY())
            postRotate(degree, targetRect.centerX(), targetRect.centerY())
            postConcat(手势矩阵)
        }
        
        canvas.drawBitmap(bitmap, targetMatrix, Paint())
    }
}

方法三

override fun onDraw(canvas: Canvas) {
    // 如果需要在一块指定Rect上绘制平移、旋转、缩放的Bitmap
    bitmap?.let { bitmap ->
        // 定义源矩形和目标矩形
        val srcRect = RectF(0f, 0f, bitmap.width.toFloat(), bitmap.height.toFloat())
        val targetRectF = RectF(targetRect)

        // 创建一个矩阵来描述从源矩形到目标矩形的变换
        val targetMatrix = Matrix().apply {
            // 设置矩形到矩形的变换,保持纵横比居中缩放
            setRectToRect(srcRect, targetRectF, Matrix.ScaleToFit.CENTER)
            // 额外的旋转变换
            postRotate(15f, targetRectF.centerX(), targetRectF.centerY())
            postConcat(手势矩阵)
        }

        // 使用变换矩阵绘制位图
        canvas.drawBitmap(bitmap, targetMatrix, Paint())
    }
}

Canvas绘制现场保护

override fun onDraw(canvas: Canvas) {
    canvas.save() // 隔离绘制现场,以下修改画布的操作不会记录
    canvas.concat(变换矩阵) // 画布绘制的所有东西,应用这个变换
    canvas.clipRect(目标区域) // 将画布裁剪到目标区域中
    canvas.clipOutRecet(目标区域) // 将画布裁剪到目标区域之外
    canvas.restore() // 恢复绘制现场,恢复修改画布之前的画布状态
    
    canvas.withSave{
        canvas.concat(变换矩阵) // 画布绘制的所有东西,应用这个变换
        canvas.clipRect(目标区域) // 将画布裁剪到目标区域中
        canvas.clipOutRecet(目标区域) // 将画布裁剪到目标区域之外
    }
}

public inline fun Canvas.withSave(block: Canvas.() -> Unit) {
    val checkpoint = save() // 返回一个Int值
    try {
        block() // 执行lambda表达式
    } finally {
        restoreToCount(checkpoint) // 从绘制栈中恢复指定Int值时的画布状态
    }
}

标签:Canvas,targetRect,val,canvas,bitmap,Bitmap,绘制
From: https://blog.csdn.net/sunshine_guo/article/details/145033413

相关文章

  • 利用Python绘制一个六边形
    使用turtle库,绘制一个六边形。‪‬‪‬‪‬‪‬‪‬‮‬‫‬‫‬‪‬‪‬‪‬‪‬‪‬‮‬‭‬‪‬‪‬‪‬‪‬‪‬‪‬‮‬‪‬‫‬‪‬‪‬‪‬‪‬‪‬‮‬‫‬‭‬‪‬‪‬‪‬‪‬‪‬‮‬‪‬‮‬‪‬‪‬‪‬‪‬‪‬‮‬‭‬‪‬‪‬‪‬‪‬‪‬‪‬‮‬‫‬‮‬‪......
  • 电路研究3——线路图绘制过程出的问题。
        因为采用的是合宙4G模组Air780E作为SIM卡的GPRS,所以需要做的是进行线路板的绘制。    遇到的问题,电源供电。之前锂电池已经实现了充电跟保护电路了。这里进行一个应用。    因为Air780E模块的供电至关重要,必须选择能够提供至少1A电流能力的电源;......
  • 前端学习openLayers配合vue3(面的绘制,至少三个点)
    我们学习了点和线的绘制,当然我们也可以绘制一个面关键代码,需要注意的一点就是面的绘制需要三维数组,线的绘制是个二维数组constpolygonLayer=newVectorLayer({source:newVectorSource(),});map.addLayer(polygonLayer);letfeature=newFeature({//......
  • 深入探索 ScottPlot.WPF:在 Windows 桌面应用中绘制精美图表的利器
    一、ScottPlot.WPF简介ScottPlot.WPF是基于ScottPlot绘图库专门为WindowsPresentationFoundation(WPF)框架量身定制的强大绘图组件。它无缝集成到WPF应用程序中,为开发者提供了一种简洁、高效的方式来可视化数据,无论是科学研究中的实验数据展示、金融领域的行情走势......
  • 前端学习openLayers配合vue3(两个坐标之间线的绘制)
    上节我们学了点的绘制,今天我们来学习一下线的绘制关键代码constlineLayer=newVectorLayer({source:newVectorSource(),});map.addLayer(lineLayer);letfeature=newFeature({//北京到上海的经纬度geometry:newLineString([[116.46,39.92],......
  • 高效工作流:用Mermaid绘制你的专属流程图;如何在Vue3中导入mermaid绘制流程图
    目录高效工作流:用Mermaid绘制你的专属流程图一、流程图的使用场景1.1、流程图flowChart1.2、使用场景二、如何使用mermaid画出优雅的流程图2.1、流程图添加图名2.2、定义图类型与方向2.3、节点形状定义2.3.1、规定语法2.3.2、不同节点案例2.4、节点连线2.5、子图与......
  • 使用 WebGL 绘制一个简单的点和原理解析
    使用WebGL绘制一个简单的点,我们需要通过WebGL的管线来进行一系列的步骤。以下是实现的详细步骤和原理解析:WebGL绘制点的基本步骤初始化WebGL上下文首先,我们需要获取WebGL上下文,这样才能进行所有的绘图操作。通常,WebGL上下文是通过<canvas>元素获取的。编......
  • 前端学习openLayers配合vue3(圆形形状的绘制)
    上节课我们学了加载了矢量图片,这节我们来学绘制圆形关键代码,第一段呢是设置圆点的操作,第二步是点击地图获取地图位置来设置圆点,ol还有很多类,各种形状的//设置圆点//letanchorLayer=newVectorLayer({//source:newVectorSource(),//});//letanchorFeat......
  • BUG:SWM32开机绘制lvgl框架下的某个自定义控件死机
    一.BUG描述现象1.画了一个关于"模式"的自定义控件,结果开机绘制总是死机。现象2.用keil进行仿真调试全速运行同样死机,但是如果在异常处加断点,然后单步调试就正常。(注:仿真调试比直接运行的速度要慢)现象3.把这个异常对象的创建代码删除,再后面加四个打印追踪,还是死机;但是删除两个......
  • LivePusherContext.createOffscreenCanvas
    LivePusherContext.createOffscreenCanvas(objectoptions)基础库2.29.0开始支持,低版本需做兼容处理。小程序插件:支持相关文档:live-pusher组件功能描述创建一个能够承接LivePusher采集纹理的离屏Canvas,客户端8.0.31版本开始支持。参数objectoptions同wx.......