webgl canvas系列——animation中基本旋转、平移、缩放(模拟冒泡排序过程)

news2025/1/11 2:19:00

文章目录

    • ⭐前言
    • ⭐canvas绘制图片
      • 💖状态保存和恢复
      • 💖移动、旋转、缩放、变形
        • 💖移动绘制一个渐变的box
        • 💖旋转
        • 💖缩放
    • ⭐模拟冒泡排序过程
    • ⭐结束

yma16-logo

⭐前言

大家好,我是yma16,本文分享webgl canvas系列——animation基本旋转、平移、缩放。
该系列往期文章
web canvas系列——快速入门上手绘制二维空间点、线、面
webgl canvas系列——快速加背景、抠图、加水印并下载图片

⭐canvas绘制图片

💖状态保存和恢复

方法作用
save()保存画布 (canvas) 的所有状态。
restore()save 和 restore 方法是用来保存和恢复 canvas 状态的,都没有参数。Canvas 的状态就是当前画面应用的所有样式和变形的一个快照。

💖移动、旋转、缩放、变形

方法作用
translate(x, y)在这里插入图片描述x *是左右偏移量,y 是上下偏移量
scale(x, y)scale方法可以缩放画布的水平和垂直的单位。两个参数都是实数,可以为负数,x 为水平缩放因子,y 为垂直缩放因子,如果比 1 小,会缩小图形,如果比 1 大会放大图形。默认值为 1,为实际大小
transform(a, b, c, d, e, f)将当前的变形矩阵乘上一个基于自身参数的矩阵

a c e b d f 0 0 1 (transform) \begin{matrix} a & c & e \\ b & d & f \\ 0 & 0 & 1 \end{matrix} \tag{transform} ab0cd0ef1(transform)

💖移动绘制一个渐变的box

使用translate移动
js代码快如下

function draw() {
    var ctx = document.getElementById("canvas").getContext("2d");
    const width=150
    const height=150
    for (var i = 0; i < 3; i++) {
      for (var j = 0; j < 3; j++) {
        ctx.save();
        ctx.fillStyle = "rgb(" + 51 * i + ", " + (255 - 51 * i) + ", 255)";
        ctx.translate(10+ j * (width+1), 10 + i * (height+1));
        ctx.fillRect(0, 0, width, height);
        ctx.restore();
      }
    }
  }

效果
translate

💖旋转

先移动原点再旋转
旋转一个正方形

  function draw() {
    const ctx = document.getElementById("canvas").getContext("2d");

    const xRectX=100
    const yRextY=100

    const width=100
    const height=100

    ctx.fillStyle = "rgba(0, 0, 200, 0.5)";
    ctx.beginPath();
    ctx.arc(xRectX, yRextY, 10, 0, Math.PI * 2, true); // 旋转中心
    ctx.fill();
    ctx.restore();
    // ctx.save();


    const translateX=xRectX+width/2
    const translateY=yRextY-height/4
  
    // left rectangles, rotate from canvas origin

    // blue rect
    ctx.fillStyle = "#0095DD";
    ctx.fillRect(xRectX, yRextY, width, height);



    ctx.fillStyle = "rgba(255, 0, 200, 0.5)";
    ctx.beginPath();
    ctx.arc(translateX, translateY, 10, 0, Math.PI * 2, true); // 旋转中心
    ctx.fill();
   
    ctx.translate(translateX, translateY); // translate back
    ctx.rotate((Math.PI / 180) * 45);
    // red rect
    ctx.fillStyle = "rgba(255,0,0,.2)";
    ctx.fillRect(0, 0, width, height);
    ctx.restore();

  }

旋转效果
蓝色的圆点移动到红色的点之后再旋转45°
rotate
移动中心 再旋转

function draw() {
    const ctx = document.getElementById("canvas").getContext("2d");

    const xRectX = 100
    const yRextY = 100

    const width = 100
    const height = 100

    ctx.fillStyle = "rgba(0, 0, 200, 0.5)";
    ctx.beginPath();
    ctx.arc(xRectX, yRextY, 10, 0, Math.PI * 2, true); // 旋转中心
    ctx.fill();
    ctx.restore();
    // ctx.save();

    // x = x + 0.5 * width
    // y = y + 0.5 * height
    const translateX = xRectX + width / 2
    const translateY = yRextY + height / 2

    // left rectangles, rotate from canvas origin

    // blue rect
    ctx.fillStyle = "#0095DD";
    ctx.fillRect(xRectX, yRextY, width, height);



    ctx.translate(translateX, translateY);
    ctx.rotate((Math.PI / 180) * 45);

    ctx.translate(-translateX, -translateY);


    // ctx.translate(translateX, translateY); // translate back

    // red rect
    ctx.fillStyle = "rgba(255,0,0,.2)";
    ctx.fillRect(xRectX, xRectX, width, height);


    ctx.fillStyle = "rgba(255, 0, 200, 0.5)";
    ctx.beginPath();
    ctx.arc(translateX, translateY, 10, 0, Math.PI * 2, true); // 旋转中心
    ctx.fill();

}

效果(红点为旋转中心)
rate_center

💖缩放

缩放文字

function draw() {
    var ctx = document.getElementById("canvas").getContext("2d");
    // mirror horizontally
    ctx.scale(2, 2);
    ctx.font = "48px serif";
    ctx.fillStyle = "rgba(0, 0, 200, 0.5)";
    ctx.fillText("csdn yma16", 0, 120);
  }

scale

⭐模拟冒泡排序过程

冒泡排序(Bubble Sort)
是一种计算机科学领域的较简单的排序算法。
它重复地走访过要排序的元素列,依次比较两个相邻的元素,如果顺序(如从大到小、首字母从Z到A)错误就把他们交换过来。走访元素的工作是重复地进行,直到没有相邻元素需要交换,也就是说该元素列已经排序完成。

vue3页面简单使用canvas模拟冒泡排序的效果

<script lang="js" setup>
import { reactive, onMounted } from 'vue';
const state = reactive({
    value: '[100,20,69,16,55,33]',
    title: '冒泡排序可视化',
    visualSortArray: [

    ]
})


function bubbleSort(arr) {
    // 冒泡排序算法,对数组进行排序,同时记录每一步操作,保存在一个数组中
    function sort() {
        // virtualArr 用来存放 每一个步内容的数组
        const virtualArr = [arr.slice()];
        console.log('virtualArr', virtualArr)
        const max = arr.length;
        for (let i = 0; i < max; i++) {
            let done = true;
            for (let j = 0; j < max - i; j++) {
                if (arr[j] > arr[j + 1]) {
                    let temp = arr[j];
                    arr[j] = arr[j + 1];
                    arr[j + 1] = temp;
                    done = false;
                    virtualArr.push(arr.slice());
                }
            };
            if (done) {
                break;
            };
        }
        return virtualArr;
    }

    // 绘画,调用一次就画出一步的图像 
    function darw(arr, count) {
        const canvas = document.getElementById('myCanvas');
        const ctx = canvas.getContext('2d');
        
        // 最大高度 400
        let maxHeight = canvas.height;
        // 每个长方形的宽度
        let width = 20;
        // 每个长方形之间的间隔
        let space = 100;

        const total = arr.reduce((p, c) => p + c, 0)

        // 清空画布
        ctx.clearRect(0, 0, canvas.width, canvas.height);
        // 设置字体
        ctx.font = "18px serif";
        // 在页面上,画出一步的内容
        for (let i = 0; i < arr.length; i++) {
            ctx.fillStyle = '#61C5FE';

            const x= i * (width + space)
            const y= maxHeight - arr[i]
            const height=Math.round((arr[i] / total)*100  + 100)
            ctx.fillRect(x, y, width, height);
            console.log('x',x)
            console.log('y',y)
            console.log('width',width)
            console.log('height',height)
            ctx.fillStyle = '#240be4';
            // 标题文字
            ctx.fillText(arr[i], x, y);
            ctx.restore();
        }

        ctx.fillText(`${count}趟排序`, 200, 100);


        ctx.fillStyle = "rgba(0, 66, 200, 0.5)";
        ctx.beginPath();
        ctx.lineTo(0, 400);
        ctx.lineTo(800, 400);
        ctx.stroke()
    }

    // 动画 
    function animation() {
        // 调用sort 方法,返回包括每一步内容的数组
        var virtualArr = sort();
        var interval = 500;
        // 遍历得到的数组,每隔500ms,调用darw 方法,画出一步内容
        virtualArr.forEach((item, index) => {
            setTimeout(() => darw(item, index + 1), index * interval);
        });

        state.visualSortArray = virtualArr
    }

    animation();
}




const sortBtn = () => {

    const arr = state.value.replace('[', '').replace(']', '').split(',').map(n => +n)
    console.log('arr', arr)
    bubbleSort(arr);
}

onMounted(()=>{
    sortBtn()
})



</script>
<template>
    <div>

        <div style="display:flex;">
            <a-card :title="state.title" style="min-width: 800px">
                <div class="input-box">

                    <div>
                        <a-input v-model:value="state.value" placeholder="请输入数组" clearable></a-input>
                    </div>
                    <div style="margin-left:50px"><a-button type="primary" @click="sortBtn">开始排序</a-button></div>
                </div>
                <div class="container-sort">
                    <div style="text-align: right;margin-right: 20px;">
                        <div v-for="(item, index) in state.visualSortArray" :key="index">
                            <div>
                                第 {{ index + 1 }} 躺排序:  {{ item.toString() }}
                            </div>
                        </div>
                    </div>

                    <canvas id="myCanvas" width="800" height="400"> </canvas>
                </div>

            </a-card>
        </div>

    </div>
</template>

<style lang="less">
.input-box {
    display: flex;
    margin-bottom: 10px;
}

.container-sort {
    height: 800px;
    border: 1px solid #dcdcdc;
}

.box {
    margin: 10px;

    .bar {
        width: 10px;
        background: #1677ff;
        border-radius: 2px;
    }

    .num {
        font-size: 18px;
    }
}
</style>

效果
sort
inscode代码块

⭐结束

本文分享到这结束,如有错误或者不足之处欢迎指出!
light

👍 点赞,是我创作的动力!
⭐️ 收藏,是我努力的方向!
✏️ 评论,是我进步的财富!
💖 最后,感谢你的阅读!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1612483.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

EMD分解

ref&#xff1a;【EMD经验模态分解-哔哩哔哩】 https://b23.tv/LfepZjU 流程&#xff1a; IMF 固有模态函数 imf1 迭代直到是一个imf 8次迭代后&#xff0c;满足是一个imf residual的得到&#xff1a;原始信号-均值&#xff08;上下包络线的均值&#xff09; imf2 为什么时序…

【公司UI自动化学习】

公司课程链接&#xff1a;https://l.jd.com/student/project/project.du?project_id697509403 公司的课程&#xff0c;是给一个学习方向。 一、 PC自动化 1&#xff09;什么项目适合 2&#xff09;PC自动化介入时间点 3&#xff09;自动化率&#xff1a; 频繁改动的&…

【Harmony3.1/4.0】笔记二

概述 列表是一种复杂的容器&#xff0c;当列表项达到一定数量&#xff0c;内容超过屏幕大小时&#xff0c;可以自动提供滚动功能。它适合用于呈现同类数据类型或数据类型集&#xff0c;例如图片和文本。在列表中显示数据集合是许多应用程序中的常见要求&#xff08;如通讯录、…

深度学习500问——Chapter08:目标检测(2)

文章目录 8.2.4 R-FCN 8.2.5 FPN 8.2.6 Mask R-CNN 8.2.4 R-FCN R-FCN 有哪些创新点 R-FCN仍然属于two-stage目标检测算法&#xff1a;RPN R-FCN Fully convolutional位置敏感得分图&#xff08;position-sentive score maps&#xff09; our region-based detector is ful…

java-Spring-入门学习-第二天(单例模式和多例模式)

目录 Bean作用域 单例模式(默认可以不写) Spring下的 AutoWired 依赖注入 JaveEE下的 Resource 依赖注入 多例模式 Bean作用域 ​在Spring框架中&#xff0c;Bean是按照作用域来创建的&#xff0c;常见的作用域有两种&#xff1a;Singleton 和 Prototype。Singleton (单例…

6.SpringBoot 日志文件

文章目录 1.日志概述2.日志作用3.使用和观察日志3.1如何观察日志3.2使用日志3.3日志级别3.4日志持久化3.5日志分割 4.日志框架4.1门面模式(外观模式)4.2 SLF4J框架介绍4.3 日志格式的说明4.3.1日志名称 5.日志颜色设置6.总结 大家好&#xff0c;我是晓星航。今天为大家带来的是…

【Java】HashMap、HashTable和ConcurrentHashMap的区别

文章目录 区别一、HashMap1.1基本定义与特性1.2工作原理与实现1.3常用方法1.4性能与优化 二、HashTable三、ConcurrentHashMap3.1基本特点3.2实现原理3.3常用方法3.4适用场景3.5性能优化 HashTable、HashMap和ConcurrentHashMap之间的区别主要体现在线程安全、继承关系与实现接…

(2024|ICLR,变分扩散模型(VDM),可学习编码器,时间相关的均值函数)DiffEnc:使用学到的编码器进行变分扩散

DiffEnc: Variational Diffusion with a Learned Encoder 公和众和号&#xff1a;EDPJ&#xff08;进 Q 交流群&#xff1a;922230617 或加 VX&#xff1a;CV_EDPJ 进 V 交流群&#xff09; 目录 0. 摘要 2. 变分扩散模型的基础 3. DiffEnc 4. 编码器和生成模型的参数化 …

Linux--地址空间

目录 看一个现象 基本概念 细节问题--理解它 1.如何理解地址空间&#xff1f; 2.为什么要有地址空间&#xff1f; 3. 进一步了解页表和写时拷贝 4.如何理解虚拟地址&#xff1f; 看一个现象 先通过一段代码&#xff0c;看一看现象 int g_val 100;int main() {printf(&quo…

Linux 认识与学习Bash——2

1 read 从键盘读取变量的值 read 后面不带变量&#xff0c;那么默认会给REPLY变量赋值 #!/bin/bash echo -n "请输入你的名字&#xff1a;" read name echo "欢迎您 $name" echo "----------------"echo -n "请输入你的名字2&#xff1a;&q…

我与深拷贝

前言 最近在掘金读到了一篇文章《Radash 能取代 Lodash&#xff1f;&#xff1f;&#xff1f;真幽默 - 掘金》&#xff0c;文章的评论区讨论起了深拷贝。"深拷贝" 我的"老朋友"&#xff0c;还记得在学习我人生中的第二道面试题的时候认识了它&#xff0c;…

CSS 画一个三角形

一、前言 在前端开发的时候&#xff0c;我们有时候会需要用到一个三角形的形状&#xff0c;比如地址选择或者播放器里面播放按钮 通常情况下&#xff0c;我们会使用图片或者svg去完成三角形效果图&#xff0c;但如果单纯使用css如何完成一个三角形呢&#xff1f; 实现过程似…

物理学视角讲解diffusion生成模型——隐扩散模型

https://zhuanlan.zhihu.com/p/692996885 https://zhuanlan.zhihu.com/p/693255617 前面两篇文章介绍了扩散过程&#xff0c;同时实现了1维、2维混合高斯扩散、逆扩散&#xff0c;通过模型预测得分函数来实现逆扩散推理。这个章节介绍工业界使用的文本生成图扩撒模型&#xff1…

基于SSM+Jsp+Mysql的多人命题系统

开发语言&#xff1a;Java框架&#xff1a;ssm技术&#xff1a;JSPJDK版本&#xff1a;JDK1.8服务器&#xff1a;tomcat7数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09;数据库工具&#xff1a;Navicat11开发软件&#xff1a;eclipse/myeclipse/ideaMaven包…

frp改造Windows笔记本实现家庭版免费内网穿透

文章目录 前言frp原理Windows服务端IP检验IP固定软件下载端口放行端口映射开机启动 NAS客户端端口查询软件下载端口检验穿透测试自启设置 Ubuntu客户端软件下载后台启动 后记 前言 之前一直用花生壳远程控制一个服务器&#xff0c;但最近内网的网络策略似乎发生了变化&#xf…

InfiniGate自研网关实现思路二

5.HTTP请求参数解析 解析 HTTP 网络请求的参数信息&#xff0c;包括&#xff1b;GET/POST&#xff0c;以及应对不同 Content-Type 类型的处理。 HTTP 接口请求的参数需要解析成可以匹配到 RPC 接口的入参信息&#xff0c;所以通常为了方便控制一般只支持 RPC 接口单个对象入参…

「sentinel」流量控制组件的应用

「sentinel」流量控制组件的应用 Sentinel版本QPS 一、初识Sentinel1、Sentinel2、Sentinel 和 Hystrix对比3、雪崩问题 二、环境搭建1、下载安装Sentinel2、微服务整合Sentinel 三、流量控制1、簇点链路2、流控设置3、流控模式直接关联链路 4、流控效果流控效果解释 四、热点限…

C#通用类库封装实战

数据库查询 特性方式获取数据库列的别名 数据库更新 使用简单工厂配置的方式

C++ stl容器stack,queue,priority_queue的底层模拟实现

目录 前言&#xff1a; 文档借鉴&#xff1a;Reference - C Reference 1.deque a.deque的结构特点&#xff1a; b.deque的迭代器结构&#xff1a; c.面试题&#xff1a; 2.stack 3.queue 4.仿函数 5.priority_queue 总结&#xff1a; 前言&#xff1a; 本篇一共简单…

【QT学习】8.qt事件处理机制,事件过滤器,自定义事件

1.qt事件处理机制 事件处理&#xff1a; 当用户移动鼠标的时候 &#xff0c;创建一个 鼠标移动事件对象 然后把这个对象放到 事件队列里面去&#xff0c;事件管理器 从队列中 取出事件&#xff0c;然后 调用其对应的事件处理函数。 多态机制&#xff1a; &#x…