实战篇:(三)项目实战Vue 3 + WebGL 创建一个简单的 3D 渲染应用

Vue 3 + WebGL 创建一个简单的 3D 渲染应用

我们将使用 Vue 3 和 WebGL 创建一个简单的 3D 渲染应用。项目将展示如何在 Vue 组件中集成 WebGL,并渲染一个旋转的立方体。

在这里插入图片描述

1. 项目准备

首先,确保你已经安装了 Node.js 和 Vue CLI。如果还没有安装,可以通过以下命令安装:

npm install -g @vue/cli

然后,创建一个新的 Vue 3 项目:

vue create vue-webgl-demo
cd vue-webgl-demo

选择 Vue 3 配置并安装项目依赖。

2. 安装依赖

为了方便 WebGL 的开发,我们可以安装 gl-matrix 库来进行矩阵和向量的数学运算:

npm install gl-matrix
3. 创建 WebGL 组件

src/components 目录下创建一个名为 WebGLCanvas.vue 的组件。代码如下:

<template><canvas ref="canvas" class="webgl-canvas"></canvas>
</template><script>
// 导入 gl-matrix 库中的 mat4 模块,用于矩阵运算
import { mat4 } from 'gl-matrix';export default {name: 'WebGLCanvas',mounted() {// 组件挂载后,初始化 WebGLthis.initWebGL();},methods: {initWebGL() {// 获取 canvas 元素并初始化 WebGL 上下文const canvas = this.$refs.canvas;const gl = canvas.getContext('webgl');// 检查 WebGL 是否初始化成功if (!gl) {console.error('无法初始化 WebGL。');return;}// 设置 canvas 的宽高为窗口的宽高canvas.width = window.innerWidth;canvas.height = window.innerHeight;// 设置视口,绘制区域与 canvas 相同gl.viewport(0, 0, canvas.width, canvas.height);// 设置背景色为黑色gl.clearColor(0.0, 0.0, 0.0, 1.0);// 清除颜色缓冲和深度缓冲gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);// 立方体的顶点数据const vertices = new Float32Array([// 前面-0.5, -0.5,  0.5, // A0.5, -0.5,  0.5, // B0.5,  0.5,  0.5, // C-0.5,  0.5,  0.5, // D// 后面-0.5, -0.5, -0.5, // E0.5, -0.5, -0.5, // F0.5,  0.5, -0.5, // G-0.5,  0.5, -0.5, // H]);// 立方体的索引数据,用于绘制面const indices = new Uint16Array([0, 1, 2, 0, 2, 3, // 前面4, 5, 6, 4, 6, 7, // 后面0, 1, 5, 0, 5, 4, // 左面2, 3, 7, 2, 7, 6, // 右面0, 3, 7, 0, 7, 4, // 上面1, 2, 6, 1, 6, 5, // 下面]);// 创建并绑定顶点缓冲区const vertexBuffer = gl.createBuffer();gl.bindBuffer(gl.ARRAY_BUFFER, vertexBuffer);// 将顶点数据写入缓冲区gl.bufferData(gl.ARRAY_BUFFER, vertices, gl.STATIC_DRAW);// 创建并绑定索引缓冲区const indexBuffer = gl.createBuffer();gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, indexBuffer);// 将索引数据写入缓冲区gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, indices, gl.STATIC_DRAW);// 定义顶点着色器const vertexShaderSource = `attribute vec4 a_Position; // 顶点位置属性uniform mat4 u_ModelViewMatrix; // 模型视图矩阵uniform mat4 u_ProjectionMatrix; // 投影矩阵void main() {// 将顶点位置转换到裁剪坐标系gl_Position = u_ProjectionMatrix * u_ModelViewMatrix * a_Position;}`;// 定义片段着色器const fragmentShaderSource = `precision mediump float; // 片段着色器精度void main() {gl_FragColor = vec4(1.0, 0.0, 0.0, 1.0); // 设置片段颜色为红色}`;// 创建并编译着色器const vertexShader = this.createShader(gl, vertexShaderSource, gl.VERTEX_SHADER);const fragmentShader = this.createShader(gl, fragmentShaderSource, gl.FRAGMENT_SHADER);// 创建着色器程序并链接着色器const program = gl.createProgram();gl.attachShader(program, vertexShader);gl.attachShader(program, fragmentShader);gl.linkProgram(program);gl.useProgram(program); // 使用该着色器程序// 绑定顶点缓冲区gl.bindBuffer(gl.ARRAY_BUFFER, vertexBuffer);// 获取顶点属性位置const a_Position = gl.getAttribLocation(program, 'a_Position');// 指定如何从缓冲区获取顶点数据gl.vertexAttribPointer(a_Position, 3, gl.FLOAT, false, 0, 0);gl.enableVertexAttribArray(a_Position); // 使能顶点属性// 设置投影矩阵和模型视图矩阵const projectionMatrix = mat4.create();const modelViewMatrix = mat4.create();// 定义透视投影mat4.perspective(projectionMatrix, Math.PI / 4, canvas.width / canvas.height, 0.1, 100);// 将模型沿 Z 轴平移mat4.translate(modelViewMatrix, modelViewMatrix, [0, 0, -3]);// 获取 uniform 变量位置const u_ProjectionMatrix = gl.getUniformLocation(program, 'u_ProjectionMatrix');const u_ModelViewMatrix = gl.getUniformLocation(program, 'u_ModelViewMatrix');// 将矩阵传入着色器gl.uniformMatrix4fv(u_ProjectionMatrix, false, projectionMatrix);gl.uniformMatrix4fv(u_ModelViewMatrix, false, modelViewMatrix);// 开始绘制this.draw(gl, program, indices.length);},createShader(gl, source, type) {// 创建着色器并编译const shader = gl.createShader(type);gl.shaderSource(shader, source);gl.compileShader(shader);// 检查着色器编译是否成功if (gl.getShaderParameter(shader, gl.COMPILE_STATUS)) {return shader;} else {console.error(gl.getShaderInfoLog(shader)); // 输出编译错误信息gl.deleteShader(shader); // 删除着色器}},draw(gl, program, numIndices) {let angle = 0; // 用于控制立方体旋转的角度const render = () => {// 清除颜色和深度缓冲gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);const modelViewMatrix = mat4.create();mat4.translate(modelViewMatrix, modelViewMatrix, [0, 0, -3]); // 将模型沿 Z 轴平移mat4.rotateY(modelViewMatrix, modelViewMatrix, angle); // 绕 Y 轴旋转// 将更新后的模型视图矩阵传入着色器const u_ModelViewMatrix = gl.getUniformLocation(program, 'u_ModelViewMatrix');gl.uniformMatrix4fv(u_ModelViewMatrix, false, modelViewMatrix);// 绘制立方体gl.drawElements(gl.TRIANGLES, numIndices, gl.UNSIGNED_SHORT, 0);angle += 0.01; // 增加角度,实现旋转requestAnimationFrame(render); // 请求下一帧};render(); // 启动渲染循环},},
};
</script><style scoped>
.webgl-canvas {width: 100%; /* canvas 占满父元素 */height: 100%; /* canvas 占满父元素 */
}
</style>
4. 使用组件

src/App.vue 中使用 WebGLCanvas 组件:

<template><div id="app"><WebGLCanvas /></div>
</template><script>
import WebGLCanvas from './components/WebGLCanvas.vue';export default {name: 'App',components: {WebGLCanvas,},
};
</script><style>
#app {margin: 0;padding: 0;overflow: hidden;
}
</style>
5. 运行项目

完成以上步骤后,可以通过以下命令运行项目:

npm run serve

打开浏览器并访问 http://localhost:8080,你将看到一个旋转的红色立方体。

在这里插入图片描述

6. 结论

通过以上步骤,我们成功创建了一个 Vue 3 + WebGL 的简单项目。你可以在此基础上扩展更多功能,比如添加纹理、交互效果或者实现复杂的 3D 场景。WebGL 的学习之路是不断探索和实践的过程,希望这篇博客能为你提供一个良好的开端!

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

解密京东详情 API 接口:获取与运用指南

一、什么是京东详情API接口&#xff1f; 京东详情API接口是京东开放平台提供的一种服务&#xff0c;允许开发者通过编程方式获取商品的详细信息。通过调用这个接口&#xff0c;你可以获取到商品的基本信息、价格、库存、评价等数据。这些数据可以帮助你更好地了解商品的情况&a…

基于WebSocket实现简易即时通讯功能

代码实现 pom.xml <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId> </dependency> <dependency><groupId>org.springframework.boot</groupId><artifa…

2024最新分别用sklearn和NumPy设计k-近邻法对鸢尾花数据集进行分类(包含详细注解与可视化结果)

本文章代码实现以下功能&#xff1a; 利用sklearn设计实现k-近邻法。 利用NumPy设计实现k-近邻法。 将设计的k-近邻法对鸢尾花数据集进行分类&#xff0c;通过准确率来验证所设计算法的正确性&#xff0c;并将分类结果可视化。 评估k取不同值时算法的精度&#xff0c;并通过…

HTML CSS 基础

HTML & CSS 基础 HTML一、HTML简介1、网页1.1 什么是网页1.2 什么是HTML1.3 网页的形成1.4总结 2、web标准2.1 为什么需要web标准2.2 Web 标准的构成 二、HTML 标签1、HTML 语法规范1.1基本语法概述1.2 标签关系 2、 HTML 基本结构标签2.1 第一个 HTML 网页2.2 基本结构标签…

uniapp 游戏 - 使用 uniapp 实现的扫雷游戏

0. 思路 1. 效果图 2. 游戏规则 扫雷的规则很简单。盘面上有许多方格,方格中随机分布着一些雷。你的目标是避开雷,打开其他所有格子。一个非雷格中的数字表示其相邻 8 格子中的雷数,你可以利用这个信息推导出安全格和雷的位置。你可以用右键在你认为是雷的地方插旗(称为标…

中华春节符号·世界品牌——粤港澳企(实)业协会商会经济合作座谈会成功举办

日前&#xff0c;一场旨在推动粤港澳三地经济深度合作的盛会——《粤港澳企&#xff08;实&#xff09;业协会商会经济合作座谈会》在广州市天河区时代TIT广场2栋801车陂社区文化中心隆重举行。此次活动由泰康之家粤园与广东经贸文化促进会联合主办&#xff0c;吸引了全球华人企…

Dubbo SpringBoot应用创建和K8S部署

推荐阅读&#xff1a;Dubbo 快速入门-CSDN博客 创建基于Spring Boot的微服务应用 以下文档将引导您从头创建一个基于 Spring Boot 的 Dubbo 应用&#xff0c;并为应用配置 Triple 通信协议、服务发现等微服务基础能力。 快速创建应用 以下是我们为您提前准备好的示例项目&am…

AI大模型开发架构设计(12)——以真实场景案例驱动深度剖析 AIGC 新时代 IT 人的能力模型

文章目录 以真实场景案例驱动深度剖析 AIGC 新时代 IT 人的能力模型1 AI 工具以及大模型会给我们带来哪些实际收益?业务研发流程环节为什么 LLM 大模型不是万能的?LLM 大模型带来实际收益 2 新时代IT人的能力模型会发生哪些变化?古典互联网架构师能力模型IT人能力模型变化 以…

这都能封!开发者行为导致Google账号关联?

从去年10月开始&#xff0c;在AI加持下&#xff0c;Google Play不断更新和迭代审查机制&#xff0c;Google Play在最近一年的时间真是杀疯了&#xff0c;封号的声音响彻整个行业&#xff0c;尤其是一些敏感品类行业。账号关联&#xff0c;恶意软件&#xff0c;欺骗行为&#xf…

小红书新ID保持项目StoryMaker,面部特征、服装、发型和身体特征都能保持一致!(已开源)

继之前和大家介绍的小红书在ID保持以及风格转换方面相关的优秀工作&#xff0c;感兴趣的小伙伴可以点击以下链接阅读~ 近期&#xff0c;小红书又新开源了一款文生图身份保持项目&#xff1a;StoryMaker&#xff0c;是一种个性化解决方案&#xff0c;它不仅保留了面部的一致性&…

贪吃蛇游戏(代码篇)

我们并不是为了满足别人的期待而活着。 前言 这是我自己做的第五个小项目---贪吃蛇游戏&#xff08;代码篇&#xff09;。后期我会继续制作其他小项目并开源至博客上。 上一小项目是贪吃蛇游戏&#xff08;必备知识篇&#xff09;&#xff0c;没看过的同学可以去看看&#xf…

Angular Count-To 项目教程

Angular Count-To 项目教程 angular-count-to Angular directive to animate counting to a number 项目地址: https://gitcode.com/gh_mirrors/an/angular-count-to 1. 项目介绍 Angular Count-To 是一个用于 AngularJS 的动画计数器指令。该指令可以在指定的时间内从…

Lfsr32

首先分析 Lfsr5 首先要理解什么是抽头点&#xff08;tap&#xff09;&#xff0c;注意到图中有两个触发器的输入为前级输出与q[0]的异或&#xff0c;这些位置被称为 tap position.通过观察上图&#xff0c;所谓抽头点指的就是第5个&#xff0c;第3个寄存器的输入经过了异或逻辑…

利用C++封装鼠标轨迹算法为DLL:游戏行为检测的利器

在现代软件开发中&#xff0c;鼠标轨迹模拟技术因其在自动化测试、游戏脚本编写等领域的广泛应用而备受青睐。本文将介绍如何使用C语言将鼠标轨迹算法封装为DLL&#xff08;动态链接库&#xff09;&#xff0c;以便在多种编程环境中实现高效调用&#xff0c;同时探讨其在游戏行…

cudnn8编译caffe过程(保姆级图文全过程,涵盖各种报错及解决办法)

众所周知,caffe是个较老的框架,而且只支持到cudnn7,但是笔者在复现ds-slam过程中又必须编译caffe,我的cuda版本是11.4,最低只支持到8.2.4,故没办法,只能编译了 在此记录过程、报错及解决办法如下; 首先安装依赖: sudo apt-get install git sudo apt-get install lib…

【IEEE独立出版 | 厦门大学主办】第四届人工智能、机器人和通信国际会议(ICAIRC 2024)

【IEEE独立出版 | 厦门大学主办】 第四届人工智能、机器人和通信国际会议&#xff08;ICAIRC 2024&#xff09; 2024 4th International Conference on Artificial Intelligence, Robotics, and Communication 2024年12月27-29日 | 中国厦门 >>往届均已成功见刊检索…

【Kubernetes① 基础】一、容器基础

目录 一、进程二、隔离与限制三、容器镜像总结参考书籍 一、进程 容器技术的兴起源于PaaS技术(平台即服务)的普及&#xff1b;Docker公司发布的Docker项目具有里程碑式的意义&#xff1b;Docker项目通过“容器镜像”解决了应用打包这个根本性难题(CloudFoundry)。 容器本身的价…

【QAMISRA】解决导入commands.json时报错问题

1、 文档目标 解决导入commands.json时报错“Could not obtain system-wide includes and defines”的问题。 2、 问题场景 客户导入commands.json时报错“Could not obtain system-wide includes and defines”。 3、软硬件环境 1、软件版本&#xff1a; QA-MISRA23.04 2、…

【电路笔记】-运算放大器多谐振荡器

运算放大器多谐振荡器 文章目录 运算放大器多谐振荡器1、概述2、施密特触发器3、运算放大器稳态多谐振荡器4、运算放大器单稳态多谐振荡器5、运算放大器双稳态多谐振荡器6、总结1、概述 本文将重点介绍通常称为多谐振荡器的配置,特别是基于运算放大器的电路。 事实上,多谐振…

AWS账号与邮箱的关系解析

在当今数字化时代&#xff0c;云计算服务的普及使得越来越多的企业和个人用户开始使用亚马逊网络服务&#xff08;AWS&#xff09;。作为全球领先的云服务平台&#xff0c;AWS为用户提供了丰富的计算、存储和数据库服务。然而&#xff0c;对于许多新用户来说&#xff0c;关于AW…