服务粉丝

我们一直在努力
当前位置:首页 > 财经 >

图形编辑器:图形和辅助线绘制的坐标问题

日期: 来源:前端西瓜哥收集编辑:西瓜

大家好,我是前端西瓜哥。今天看看绘制图形和辅助线时,坐标转换的一些注意点。

项目地址,欢迎 star:

https://github.com/F-star/suika

线上体验:

https://blog.fstars.wang/app/suika/

先回顾一下之前讲的视口坐标和场景坐标的转换关系。

图形编辑器:场景坐标、视口坐标以及它们之间的转换

视口转场景:

function viewportCoordsToSceneCoords(x, y, scrollX, scrollY, zoom) {
  return {
  x: scrollX + x / zoom,
  y: scrollY + y / zoom
  }
}

场景转视口:

function sceneCoordsToViewportCoords(x, y, scrollX, scrollY, zoom) {
  return {
  x: (x - scrollX) * zoom,
  y: (y - scrollY) * zoom
  };
}

图形的绘制

场景很大,但能画的范围其实就视口大小。所以,我们需要将使用了场景坐标的图形的位置,转换为视口坐标,再绘制。

有一样非常低效的做法,就是生成一个非常大的 Canvas 画布,将其中的图形全部都画出来,然后用一个 div 容器装下,然后给 div 设置 overflow: scroll。然后调整一下 div 的 scrollLeft 和 scrollTop 就好。不推荐,效率很差。

对于图形我们的做法是在绘制图形前,先做矩阵变换,让之后绘制的所有像素都自动做一个转换,不用自己一个个手动转换。

有的朋友看着前面的 sceneCoordsToViewportCoords 方法,有:

viewportX = (sceneX - scrollX) * zoom;

于是认为 ctx 的变换对应的写法是这样的:

ctx.translate(-viewport.x, -viewport.y);
ctx.scale(zoom, zoom);

// 绘制各种图形
// ...

这个写法思路是对的,但细节有错误。因为 ctx.scale 的缩放中心因为前面的ctx.tranlate(0, 0) 变成了 (-viewport.x, -viewport.y)

正确的写法其实是缩放时调整一下缩放中心,缩放后再移回去,即:

ctx.translate(-viewport.x, -viewport.y);
ctx.translate(viewport.x, viewport.y);
ctx.scale(zoom, zoom);
ctx.translate(-viewport.x, -viewport.y);

然后你会发现,第一行和第二行抵消了,于是化简得到:

ctx.scale(zoom, zoom);
ctx.translate(-viewport.x, -viewport.y);

// 绘制各种图形
// ...

辅助线的绘制

上面的效果,是无差别给之后绘制的所有图形做缩放。也就是说,zoom 变大时,线宽(ctx.lineWidth)也会跟着变大。

图形编辑器要绘制的除了图形外,还有非常重要的一样东西:辅助线

(辅助线的坐标我们也是用场景坐标系的)

对于辅助线,我们希望 zoom 改变时,还能让线宽保持原来的 1px,还有让控制点的尺寸不变,如下图效果:

缩放功能演示

解决方案是,我们自己算辅助线上的点在视口坐标的位置,不借助 ctx.scalectx.translate

// 变换矩阵重置
ctx.setTransform(1, 0, 0, 1, 0, 0);
// 计算视口坐标系下的坐标值
const {x, y} = sceneCoordsToViewportCoords(rotationX, rotationY, viewport.x, viewport.y, zoom)

首先用 ctx.setTransform 将变换矩阵重置,将之前 ctx.scale 等造成的影响消除掉。

然后就是用前面写好的 sceneCoordsToViewportCoords 方法转换一下,得到视口坐标系下的位置,然后进行绘制即可。

其实就是局部做坐标系转换,比如坐标会转换、线宽不转换。其实也有另一种思路,就是让线宽除以 zoom,或尺寸除以 zoom,都可以。

结尾

场景坐标和视口坐标转换,贯穿于整个编辑器项目,还是很重要的,要好好消化。

我是前端西瓜哥,欢迎关注我,学习更多前端知识。



相关阅读,

图形编辑器:标尺功能的实现

图形编辑器:旋转选中的元素

图形编辑器:场景坐标、视口坐标以及它们之间的转换

图形编辑器——矩形选区是如何实现选中多个图形的?

在容器内显示图片的五种方案:contain、cover、fill、none、scale-down

计算机图形学:变换矩阵

快速检索碰撞图形:四叉树碰撞检测

Canvas 性能优化:脏矩形渲染

如何在 Canvas 上实现图形拾取?

来,教你开发一款图形编辑器


关注公众号,后台回复 「字节」,即可获得字节前端面试资料

相关阅读

  • 回顾 2022 / 展望 2023

  • 其实不想写的,因为今年挺糟心的,下不动笔。但快过年了,思前想后,最后还是写一下吧。回顾 2022先过一下 2022 立下的 flag。1、每个月至少三篇文章,再尝试看能不能投稿。这个对我
  • 图形编辑器:工具管理和切换

  • 大家好,我是前端西瓜哥。今天我们看看对于一款图形编辑器,应该怎么去实现工具,比如绘制矩形、选中工具,以及如何去管理它们的。项目地址,欢迎 star:https://github.com/F-star/sui
  • WebSocket 入门:简易聊天室

  • 大家好,我是前端西瓜哥,今天我们用 WebSocket 来实现一个简单的聊天室。WebSocket 是一个应用层协议,有点类似 HTTP。但和 HTTP 不一样的是,它支持真正的全双工,即不仅客户端可以
  • 在 VSCode 中像写 TypeScript 一样写 JavaScript

  • 大家好,我是前端西瓜哥。我们在 VSCode 编辑器中编写 js 代码,是会提供类型提示的。VSCode 会推断一个变量是什么类型,并在你输入内容的时候,提供对应的 API 属性或方法补全。如
  • 一起学 pixijs(2):修改图形属性

  • 大家好,我是前端西瓜哥。我们做动画、游戏、编辑器,需要根据用户的交互等操作,去实时地改变图形的属性,比如位置,颜色等信息。今天西瓜哥带大家来看看在 pixijs 怎么修改图形的属
  • Amazing!如何根据背景色自动切换黑白文字?

  • 在项目中,经常会碰到背景色不确定的场景,为了让内容文字足够清晰可见,文字和背景之间需要有足够的对比度。换句话说,当背景是深色时,文字为白色,当背景是浅色时,文字为黑色,就像这样
  • 巧用视觉障眼法,还原 3D 文字特效

  • 最近群里有这样一个有意思的问题,大家在讨论,使用 CSS 3D 能否实现如下所示的效果:这里的核心难点在于,如何利用 CSS 实现一个立体的数字?CSS 能做到吗?不是特别好实现,但是,如果仅
  • 那些炫酷的 CSS 文字效果之诗词《兔》

  • 不知不觉已经迈入2023年,今年是兔年,想到兔子就会联想到玉兔,中秋,胡萝卜,兔子不吃窝边草,这就越扯越远了,今天的主题是用纯CSS来实现各种不错的文字效果,文字则摘录古诗词中有关《
  • 今晚 B 站直播预告 - 《你所不知道的CSS》

  • 扫码直达 B 站直播间,7 号晚 20:00 点准时开始,预计时长 1 小时,感兴趣来听听:Hi 各位小伙伴,本周六(2023/01/07)晚 20:00 点,我将在 Bilibili 给大家带来一场《你所不知道的CSS》
  • 不负时光,Coco 的 2022 年终总结

  • 从来没有写过年终总结,回首 2022,感觉有很多令人难忘的瞬间,觉得今年很有必要记录一下。在回家的列车上,回忆一整年的的经历,有挣扎、有坚持、有迷茫、有喜悦,最终也有收获。用文

热门文章

  • “复活”半年后 京东拍拍二手杀入公益事业

  • 京东拍拍二手“复活”半年后,杀入公益事业,试图让企业捐的赠品、家庭闲置品变成实实在在的“爱心”。 把“闲置品”变爱心 6月12日,“益心一益·守护梦想每一步”2018年四

最新文章

  • 图形编辑器:图形和辅助线绘制的坐标问题

  • 大家好,我是前端西瓜哥。今天看看绘制图形和辅助线时,坐标转换的一些注意点。项目地址,欢迎 star:https://github.com/F-star/suika线上体验:https://blog.fstars.wang/app/suika
  • 回顾 2022 / 展望 2023

  • 其实不想写的,因为今年挺糟心的,下不动笔。但快过年了,思前想后,最后还是写一下吧。回顾 2022先过一下 2022 立下的 flag。1、每个月至少三篇文章,再尝试看能不能投稿。这个对我
  • 图形编辑器:工具管理和切换

  • 大家好,我是前端西瓜哥。今天我们看看对于一款图形编辑器,应该怎么去实现工具,比如绘制矩形、选中工具,以及如何去管理它们的。项目地址,欢迎 star:https://github.com/F-star/sui
  • WebSocket 入门:简易聊天室

  • 大家好,我是前端西瓜哥,今天我们用 WebSocket 来实现一个简单的聊天室。WebSocket 是一个应用层协议,有点类似 HTTP。但和 HTTP 不一样的是,它支持真正的全双工,即不仅客户端可以
  • 在 VSCode 中像写 TypeScript 一样写 JavaScript

  • 大家好,我是前端西瓜哥。我们在 VSCode 编辑器中编写 js 代码,是会提供类型提示的。VSCode 会推断一个变量是什么类型,并在你输入内容的时候,提供对应的 API 属性或方法补全。如
  • 一起学 pixijs(2):修改图形属性

  • 大家好,我是前端西瓜哥。我们做动画、游戏、编辑器,需要根据用户的交互等操作,去实时地改变图形的属性,比如位置,颜色等信息。今天西瓜哥带大家来看看在 pixijs 怎么修改图形的属