当前位置: 首页 > 编程日记 > 正文

用canvas实现一个vue弹幕组件

看B站时,对弹幕的实现产生了兴趣,一开始想到用css3动画去实现,后来感觉这样性能不是很好,查了下资料,发现可以用canvas实现,于是就摸索着写了一个简单的弹幕。

弹幕功能

  1. 支持动态添加弹幕
  2. 弹幕不重叠
  3. 自定义弹幕颜色

效果图

demo 

源码地址


前端框架选了比较熟悉的vuejs

弹幕滚动的基本思路就是通过定时器不断地改变弹幕的位置,时时重绘画布。

实现步骤

先加入一个canvas标签,这里有个注意点,关于设备像素比对canvas的影响,会出现绘图模糊。

canvas 绘图模糊问题

<canvas width="600" height="600"></canvas> // 如果单纯这样写,canvas会出现模糊<canvas width="600" height="600" style="width: 300px;height: 300px"></canvas>//为了不出现模糊,需要设置canvas的css宽高为上下文宽高的1/devicePixelRatio,
本文是对于devicePixelRatio:2的设备设置的,该值可从window.devicePixelRatio取得。
<canvas ref="hiddenCanvas" width="0" height="0" style="display: none"></canvas> 
// 后面会用到复制代码

我们先定义一个数组来存放弹幕数据,一条弹幕信息,包括文本内容,x,y坐标位置,颜色,速度(可以是随机或者固定,为了计算简单,我们这里采用了固定的速度)

var dmArr = [];
var gap = 80; // 弹幕的上下间距
var hiddenCanvas = this.$refs.hiddenCanvas;// 增加弹幕的方法
function pushDm(text, color) {let y = getY(); // 先确定跑道let x = 600; // 初始x坐标为canvas的右边界let delayWidth = 0; // 同跑道for (let i = 0, len = dmArr.length; i < len; i++) {let dm = dmArr[i];if (y === dm.y) { // 如果是同跑道,则往后排,设置一定的间隔,保证弹幕不会重叠;delayWidth += Math.floor(hiddenCanvas.getContext('2d').measureText(dm.text).width * 4 + 50);}   }dmArr.push({text: text,x: x + delayWidth,y: y,speed: 8,color: color || getColor()});
}
// 随机获得y坐标
function getY() {let range = Math.floor(600 / gap); // 跑道数量return Math.floor(Math.random() * range + 1) * gap;
}
// 随机获得颜色
function getColor() {return `${Math.floor(Math.random() * 16777215).toString(16)}`;
}// 写一个for循环,初始化30条弹幕
for (let i = 0; i < 30; i++) {pushDm(`It's barrage ${i}`);
}复制代码

接下来设置一个20ms的定时器,实现弹幕滚动效果

var timer = null;
var ctx = this.$refs.canvas.getContext('2d');
function start(){timer = setInterval(() => {ctx.clearRect(0, 0, 600, 600); // 每次需要清空画布ctx.save();ctx.font = '30px Microsoft YaHei'; // 这里需要把字体大小设为需要显示的css大小的2倍(devicePixelRatio为2时)if (!dmArr.length) stop(); // 如果没有新弹幕了,就停止计时器for (let i = 0, len = this.dmArr.length; i < len; i++) {let dm = dmArr[i];let overRange = -ctx.measureText(dm.text).width * 2;dm.x -= dm.speed;if (dm.x < overRange) {dmArr.splice(i, 1); // 弹幕在画布中不可见时,从数组中移除该项continue;}ctx.fillStyle = `#${dm.color}`;ctx.fillText(dm.text, dm.x, dm.y);}ctx.restore();}, 20);
}
function stop() {clearInterval(timer);ctx.clearRect(0, 0, 600, 600);
}复制代码

我们还需要一个输入框,来实现手动添加弹幕功能

<input type="text" @keyup.enter="sent" v-model="dmInput" maxlength="20">
<button type="button" @click="sent">发表</button>var dmInput = '';
var color = ''; // 可自定义弹幕的颜色
function sent() {if (!dmInput) return;stop();pushDm(dmInput, color);start();dmInput = '';
}复制代码

有待改进的地方和疑问?

  1. 速度不恒定时,怎么保持弹幕不重叠
  2. 视频弹幕是根据弹幕发送时间点来定位到视频的每一帧?如何实现?


转载于:https://juejin.im/post/5bfe3f7c6fb9a04a0c2e250b

相关文章:

系统架构的过程 浮现式设计

系统架构如果设计之初就设计错了&#xff0c;那么必然是南辕北辙。 很多人做系统设计总是东一下&#xff0c;西一下&#xff0c;杂乱无章&#xff0c;想到那是那&#xff0c;然后系统的边界很大&#xff0c;总会有疏漏。 那么系统架构应该怎么设计呢&#xff1f; 首先来说分…

【Python】Listbox组件 Scrollbar组件 Scale组件

Listbox组件 在选项特别多的时候&#xff0c;Listbox是以列表的形式显示出来&#xff0c;并支持滚动条操作&#xff0c;所以在对于需要提供大量选项的情况下会更适用。 from tkinter import * root Tk() theLB Listbox(root,setgridTrue) theLB.pack() for item in ["…

Hive 按某列的部分排序 以及 删列操作

Hive 按某列的部分排序 以及 删列操作 脑袋果然还是智商不足。 涉及到的小需求&#xff1a; 某个表test 有一列 tc&#xff1a; a字符串b字符串c字符串 拼接组成把test表&#xff0c;按b字符串排序 输出遇到的问题&#xff1a; select 里面必须包含 order by 的列按b字符串排序…

docker 安装 RabbitMQ

1、镜像中国&#xff08;http://www.docker-cn.com/registry-mirror&#xff09;&#xff1a;直接使用https://hub.docker.com下载镜像比较慢&#xff0c;使用镜像中国加速 使用例子&#xff1a;$ docker pull registry.docker-cn.com/library/rabbitmq:3.6-management 2、拉取…

编程容易犯的错

1.数字 比如分页默认从第0页开始&#xff0c;你不了解&#xff0c;写个1&#xff0c;这样数据也出来&#xff0c;但是就是少了第一条&#xff0c;这种错误比较难发现。 写代码对于数字需要敏感&#xff0c;不懂一定要搞懂。 2.参数 多参数做缓存拼凑一个key&#xff0c;之前…

深圳杯---无线回传拓扑规划

B题-无线回传拓扑规划&#xff08;3人完成&#xff09; 背景介绍 在城区建设基站&#xff0c;传输光纤部署最后一公里的成本高&#xff0c;光纤到站率低&#xff0c;全球综合来看低于60%&#xff1b;如果使用微波传输&#xff0c;由于微波只能在LOS&#xff08;视距&#xff0…

Jmeter脚本 GUI和非GUI启动方式

2019独角兽企业重金招聘Python工程师标准>>> 1&#xff0e;下载Jmeter 地址&#xff1a;http://jmeter.apache.org/download_jmeter.cgi 2&#xff0e;启动jmeter 运行bin/jmeter.bat 3&#xff0e;添加线程组 在TestPlan节点上右键&#xff0c;Add-->Threads(U…

前端效果参考地址

今天项目内容基本完善&#xff0c;没什么事情&#xff0c;就找了一些插件和好用的css动画&#xff0c;下面将一些链接地址分享出来 1、如果需要写阴影、圆角、渐变、弹性盒子等&#xff0c;请参考一下方式&#xff1a; 点击 2、轮播图、全屏滚动等动画&#xff1a; swiper效果 …

随机变量的数字特征(数学期望,方差,协方差与相关系数)

戳这里&#xff1a;概率论思维导图 &#xff01;&#xff01;&#xff01; 数学期望 离散型随机变量的数学期望 &#xff08;这里要求级数绝对收敛&#xff0c;若不绝对收敛&#xff0c;则E(X)不存在&#xff09; 如果有绝对收敛&#xff0c;则有 &#xff0c;其中 连续型…

Spring @bean冲突解决方案

引用2个jar都实现了相同的bean注入&#xff0c;这个是feign的Level Bean public Level feignLoggerLevel() {return Level.FULL; } 这样报错: escription:xxx required a single bean, but 2 were found:- feignLoggerLevel: defined by method feignLoggerLevel in class p…

javascript中实例方法与类方法的区别

在javascript中&#xff0c;类有静态属性和实例属性之分&#xff0c;也有静态方法和实例方法之分 类属性&#xff08;静态属性&#xff09;&#xff1a;通过类直接访问&#xff0c;不需要声明类的实例来访问 类方法&#xff08;静态方法&#xff09;&#xff1a;通过类直接访问…

vue 集成富文本tinymce

开发环境 1. vscode开发语言 1. vue 2. javaScript插件安装 1. npm install tinymce -S 2. 可以使用里面的文件&#xff0c; 下载后可以在node_modules 里面查看如下未目录结构3. 可以将整个结构拷在static里面&#xff0c;为了节省打包后的文件大小可以将tinymce.min.js以cdn方…

c语言中如何设计和编写一个应用系统?

C程序中,如何设计和编写一个应用系统?一、 C语言文件的操作1、 文件操作的基本方法&#xff1a;C语言将计算机的输入输出设备都看作是文件。例如&#xff0c;键盘文件、屏幕文件等.向屏幕输出一个信息&#xff0c;例如“Hello”是#include.h>int main(){printf("Hello…

深圳杯---人才吸引力评价模型研究

人才吸引力评价模型研究 在世界各国和全国各地都加大争夺人才的背景下&#xff0c;一个城市要保持其竞争活力和创新力&#xff0c;必须与时俱进地但不盲目地调整相关人才吸引政策。2018年深圳市将加大营商环境改革力度作为一项重要工作&#xff0c;以吸引更多优秀的高新企业和…

不写容易出错的代码

下面2段代码都是完成商品名称的更新&#xff0c;只是第一种情况数据源是list第二种是map 第一代代码是从List里获取第0个 entity.setProduct_name(productList.get(0).getName()); 第二段代码从map里获取键值 entity.setProduct_name(productMap.get(pid).getName())); 如果…

【Vue】IView之table组件化学习(二)

最基本的绑定table是这样的&#xff0c;需要columns和data两个属性。 <template><Card><h4>表格栗子</h4><Table :columns"cols" :data"stuList"></Table></Card> </template><script> export defa…

show-busy-java-threads查找CPU占用高

背景&#xff1a;需要查找线上CPU占用过高的Java线程在做什么。 可以使用top命令找出占CPU高的进程 #top 然后按shiftC 按CPU占比排序 然后把进程中占比高的线程id找出来&#xff0c;这个是常见的套路&#xff0c;但是这样做比较繁琐。 可以使用show-busy-java-threads工具…

了解机器学习的八大专业术语

转自&#xff1a;https://www.sohu.com/a/217453268_178466 1 自然语言处理 自然语言处理对于许多机器学习方法来说是一个常用的概念&#xff0c;它使得计算机理解并使用人所读或所写的语言来执行操作成为了可能。 自然语言处理最重要的最有用的实例&#xff1a; ① 文本分类…

34.TokenInterceptor防止表单重复提交

转自&#xff1a;https://wenku.baidu.com/view/84fa86ae360cba1aa911da02.html 由于某些原因&#xff0c;用户在进行类似表单提交的操作后&#xff0c;以为表单未被提交&#xff0c;会进行多次的重复提交。为了避免用户多次提交给服务器带来负荷。我们会对表单提交这样的操作进…

使用arthas采集火焰图

火焰图是用图形化的方式来展现profiler工具采集的性能数据&#xff0c;对数据进行统计和分析&#xff0c;方便找出性能热点。 现在我们使用arthas采集JVM的火焰图。 1.首先你需要安装arthas 说是安装其实就是下载解压&#xff0c;arthas是不需要安装的。 下载 — Arthas 3.5…

sudo配置文件详解及实战

2019独角兽企业重金招聘Python工程师标准>>> 安装NGINX之后每次都需要切换ROOT用户做配置文件修改和启动&#xff0c;为了加强安全&#xff0c;ROOT用户一般是不允许直接提供给应用开发人员或者运维人员的&#xff0c;所以需要提供一种方法可以一般用户执行ROOT用户…

Centos中文输入法安装以及切换

鼓捣鼓捣&#xff08;我是一只菜鸟&#xff09;&#xff0c;终于在我的Centos上面装上我的大中华输入法了&#xff0c;哈哈哈哈下面就简单描述下安装过程吧&#xff01;&#xff01;&#xff01;centos6.5用yum安装中文输入法打开终端&#xff0c;进入root用户&#xff08;命令…

【MATLAB】矩阵信息的获取

1、矩阵结构 矩阵的结构是指矩阵子元素的排列方式。 函数名称函数功能isempty(A)检测矩阵是否为空isscalar(A)检测矩阵是否是单元素的标量矩阵isvector(A)检测矩阵是否是只具有一行或一列元素的一维向量issparse(A)检测数组是否是系数矩阵 返回1表示该矩阵是某一特定类型的矩…

Android Gradle Plugin 源码解析(上)

一、源码依赖 本文基于: android gradle plugin版本&#xff1a; com.android.tools.build:gradle:2.3.0 gradle 版本&#xff1a;4.1 Gradle源码总共30个G&#xff0c;为简单起见&#xff0c;方便大家看源码&#xff0c;此处通过gradle依赖的形式来查看源码&#xff0c;依赖源…

Guice系列之用户指南(七)

原文地址&#xff1a;https://code.google.com/p/google-guice/wiki/ToConstructorBindings Constructor Bindings&#xff08;构造器绑定&#xff09;&#xff1a;在父类型上绑定子类实现的构造函数。 贴代码&#xff1a; 12345678910111213141516171819202122232425262728293…

Linux系统火焰图

CentOS7.8 安装perf #yum install perf 执行perf 执行perf record 命令&#xff0c;记录该PID的行为 #perf record -a -g -p 14851 -- sleep 30 --30秒后退出 需要注意后面生成svg图片的所有命令要和当前perf在同一目录&#xff0c;不然会报错。 #perf report 安装git …

深圳杯---垃圾焚烧厂的经济补偿问题

垃圾围城是世界性难题&#xff0c;在今天的中国显得尤为突出。2012年全国城市生活垃圾清运量达到1.71亿吨&#xff0c;比2010年增长了1300万吨。数据显示&#xff0c;目前全国三分之二以上的城市面临垃圾围城问题&#xff0c;垃圾堆放累计侵占土地75万亩。因此&#xff0c;垃圾…

make -j8以及linux下查看cpu的核数

# 总核数 物理CPU个数 X 每颗物理CPU的核数 # 总逻辑CPU数 物理CPU个数 X 每颗物理CPU的核数 X 超线程数# 查看物理CPU个数 cat /proc/cpuinfo| grep "physical id"| sort| uniq| wc -l# 查看每个物理CPU中core的个数(即核数) cat /proc/cpuinfo| grep "cpu …

IDEA2021.3.2拉取maven报错maven-default-http-blocker解决方法

因为IDEA2021.3.2 的Maven是3.8.1后&#xff0c;mvn编译的时候总是提示拉不到依赖&#xff0c;报错如下&#xff1a; Could not validate integrity of download from http://0.0.0.0/... 因为使用HTTP协议下载依赖&#xff0c;可能会导致中间人攻击。 所以Maven 3.8.1就禁止…

2013高教社杯---B碎纸片的拼接复原

破碎文件的拼接在司法物证复原、历史文献修复以及军事情报获取等领域都有着重要的应用。传统上&#xff0c;拼接复原工作需由人工完成&#xff0c;准确率较高&#xff0c;但效率很低。特别是当碎片数量巨大&#xff0c;人工拼接很难在短时间内完成任务。随着计算机技术的发展&a…