Canvas 文字碰撞检测并抽稀的方法


Posted in HTML / CSS onMay 27, 2019

需求背景

一般在做地图相关的需求是才会用到文字抽稀,我也是在为公司的地图引擎实现一个功能时才实现了该方法,在这里将其简化了,就在普通的 Canvas 上进行操作,并没有引入地图概念

效果

Canvas 文字碰撞检测并抽稀的方法

碰撞检测

计算文字在 canvas 中所占据的范围

// 计算文字所需的宽度
var p = {
  x: 10,
  y: 10,
  name: "测试文字"
};
var measure = ctx.measureText(p.name);
// 求出文字在 canvas 画板中占据的最大 y 坐标
var maxX = measure.width + p.x;
// 求出文字在 canvas 画板中占据的最大 y 坐标
// canvas 只能计算文字的宽度,并不能计算出文字的高度。所以就利用文字的宽度除以文字个数计算个大概
var maxY = measure.width / p.name.length + p.y;

var min = { x: p.x, y: p.y };
var max = { x: maxX, y: maxY };
// bounds 为该文字在 canvas 中所占据的范围。
// 在取点位坐标作为最小范围时,textAlign、textBaseline 按照以下方式设置会比较准确。
// 如设置在不同的位置展示,范围最大、最小点也需进行调整
// ctx.textAlign = "left";
// ctx.textBaseline = "top";
var bounds = new Bounds(min, max);

Bounds 范围对象

/**
 * 定义范围对象
 */
function Bounds(min, max) {
  this.min = min;
  this.max = max;
}

/**
 * 判断范围是否与另外一个范围有交集
 */
Bounds.prototype.intersects = function(bounds) {
  var min = this.min,
    max = this.max,
    min2 = bounds.min,
    max2 = bounds.max,
    xIntersects = max2.x >= min.x && min2.x <= max.x,
    yIntersects = max2.y >= min.y && min2.y <= max.y;

  return xIntersects && yIntersects;
};

检测

// 每次绘制之前先与已绘制的文字进行范围交叉检测
// 如发现有交叉,则放弃绘制当前文字,否则绘制并存入已绘制文字列表
for (var index in _textBounds) {
  // 循环所有已绘制的文字范围,检测是否和当前文字范围有交集,如果有交集说明会碰撞,则跳过该文字
  var pointBounds = _textBounds[index];
  if (pointBounds.intersects(bounds)) {
    return;
  }
}

_textBounds.push(bounds);
ctx.fillStyle = "red";
ctx.textAlign = "left";
ctx.textBaseline = "top";
ctx.fillText(p.name, p.x, p.y);

示例、代码地址

示例地址:示例

具体可查看完整代码:Github 地址

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

HTML / CSS 相关文章推荐
Web前端绘制0.5像素的几种方法
Aug 11 HTML / CSS
一款css实现的鼠标经过按钮的特效
Sep 11 HTML / CSS
基于Jquery和Css3代码制作可以缩放的搜索框
Nov 19 HTML / CSS
html+css3实现的登录界面
Dec 09 HTML / CSS
HTML5 Canvas实现玫瑰曲线和心形图案的代码实例
Apr 10 HTML / CSS
HTML5 Canvas概述
Aug 26 HTML / CSS
HTML5之SVG 2D入门7—SVG元素的重用与引用
Jan 30 HTML / CSS
用HTML5实现网站在windows8中贴靠的方法
Apr 21 HTML / CSS
使用HTML5拍照示例代码
Aug 06 HTML / CSS
HTML5中原生的右键菜单创建方法
Jun 28 HTML / CSS
详解canvas绘制网络字体几种方法
Aug 27 HTML / CSS
HTML5中的DOCUMENT.VISIBILITYSTATE属性详解
May 07 HTML / CSS
利用canvas实现图片下载功能来实现浏览器兼容问题
May 31 #HTML / CSS
HTML5印章绘制电子签章图片(中文英文椭圆章、中文英文椭圆印章)
Jun 03 #HTML / CSS
详解利用canvas实现环形进度条的方法
Jun 12 #HTML / CSS
Html5 实现微信分享及自定义内容的流程
Aug 20 #HTML / CSS
前端canvas动画如何转成mp4视频的方法
Jun 17 #HTML / CSS
详解FireFox下Canvas使用图像合成绘制SVG的Bug
Jul 10 #HTML / CSS
canvas实现有递增动画的环形进度条的实现方法
Jul 10 #HTML / CSS
You might like
PHP 自定义错误处理函数trigger_error()
2013/03/26 PHP
PHP在引号前面添加反斜杠(PHP去除反斜杠)
2013/09/28 PHP
Thinkphp3.2.3分页使用实例解析
2016/07/28 PHP
Ext.FormPanel 提交和 Ext.Ajax.request 异步提交函数的区别
2009/11/12 Javascript
js中格式化日期时间型数据函数代码
2010/11/08 Javascript
不用锚点也可以平滑滚动到页面的指定位置实现代码
2013/05/08 Javascript
如何通过js实现图片预览功能【附实例代码】
2016/03/30 Javascript
JS命令模式例子之菜单程序
2016/10/10 Javascript
React 使用browserHistory项目访问404问题解决
2018/06/01 Javascript
基于Vue实现电商SKU组合算法问题
2019/05/29 Javascript
解决vue的过渡动画无法正常实现问题
2019/10/31 Javascript
Vue的props父传子的示例代码
2020/05/20 Javascript
vue使用svg文件补充-svg放大缩小操作(使用d3.js)
2020/09/22 Javascript
[04:59]2018DOTA2亚洲邀请赛 4.7 Mineski夺冠时刻
2018/04/09 DOTA
使用Python判断IP地址合法性的方法实例
2014/03/13 Python
Python控制多进程与多线程并发数总结
2016/10/26 Python
Python数据可视化库seaborn的使用总结
2019/01/15 Python
WxPython建立批量录入框窗口
2019/02/27 Python
python3.7将代码打包成exe程序并添加图标的方法
2019/10/11 Python
Python socket服务常用操作代码实例
2020/06/22 Python
scrapy结合selenium解析动态页面的实现
2020/09/28 Python
html5中如何将图片的绝对路径转换成文件对象
2018/01/11 HTML / CSS
爱尔兰家电数码商城:Currys PC World爱尔兰
2016/07/23 全球购物
泰国办公用品购物网站:OfficeMate
2018/02/04 全球购物
Coltorti Boutique官网:来自意大利的设计师品牌买手店
2018/11/09 全球购物
波兰在线儿童和婴儿用品零售商:pinkorblue
2019/06/29 全球购物
辅导员评语
2014/05/04 职场文书
处级领导干部四风问题自我剖析材料
2014/09/29 职场文书
2015年学校信息技术工作总结
2015/05/25 职场文书
评估“风险”创业计划的几大要点
2019/08/12 职场文书
详解CSS开发过程中的20个快速提升技巧
2021/05/21 HTML / CSS
Python利用folium实现地图可视化
2021/05/23 Python
Java设计模式之代理模式
2022/04/22 Java/Android
SQL Server中搜索特定的对象
2022/05/25 SQL Server
mysql中关键词exists的用法实例详解
2022/06/10 MySQL
纯CSS实现一个简单步骤条的示例代码
2022/07/15 HTML / CSS