当前位置: 首页 > news >正文

网站建设需要哪些语言展示型网站有哪些

网站建设需要哪些语言,展示型网站有哪些,小程序开发平台找哪家好,自己网站做反链【趟坑记录】d3.zoom()的正确使用姿势 d3.v7 文章目录 【趟坑记录】d3.zoom()的正确使用姿势 d3.v7问题重现原因分析解决方案放缩平移写法特殊修改transform函数的写法 总结 在开发一个D3应用的时候遇到了一个 zoom相关的问题,记录解决思路与方案 问题重现 最近在…

【趟坑记录】d3.zoom()的正确使用姿势 @d3.v7

文章目录

  • 【趟坑记录】`d3.zoom()`的正确使用姿势 @d3.v7
      • 问题重现
      • 原因分析
      • 解决方案
        • 放缩平移写法
        • 特殊修改`'transform'`函数的写法
      • 总结

在开发一个D3应用的时候遇到了一个 zoom相关的问题,记录解决思路与方案

问题重现

最近在开发一个D3应用的时候遇到了一个zoom相关的问题,应用里有一个功能叫全景聚焦。我们都知道画布由两个标签组成(见实现autoZoom(),画布自适应放缩并居中@D3.js-v5),最外层的是固定视口<svg>,一般将zoom事件绑定在<svg>上;内层是具体的画布,是一个<g>标签,在<svg>中的放缩与平移操作都作用在<g>上,修改<g>transform属性。这么做是为了避免用户将<svg>元素拖动到窗口之外后丢失拖动焦点,无法将其拖回。而如果使<svg>不动,<g>被拖动,那么拖动焦点就不会丢失,用户将<g>元素移动至视口外后,还能将其拖回来。

我之前习惯这么写拖动平移:

const svg = d3.select('#viewport').attr('width', width).attr('height', height)
const g = svg.append('g').attr('id', 'container').attr('width', width).attr('height', height)svg.call(d3.zoom().on('zoom', (e) => {const transform = `translate(${e.transform.x},${e.transform.y}) scale(${e.transform.k})`g.attr('transform', transform)})
)

在一些业务场景中,往往需要对<g>元素进行特定的平移与放缩。如:自动缩放至视口中央,放大至当前的1.5倍。然而,在其他直接地方修改了<g>‘transform’属性后,如:

const offsetX = 10;
const offsetY = 10;
g.attr('transform',`translate(${offsetX},${offsetY})`

,问题就出现了,如下:

bugreproduce

可以看到,在设置了特定的'transform'后,再进行拖动,会出现瞬移。

原因分析

因为监听的zoom事件是通过e.transform来进行放缩的。而在修改<g>元素的‘transform’属性为一个特定值后,再进行拖动,会从上一次的e.tranform值开始修改,因此会出现错误。

举例说明:

  1. 用户拖动,e.transform的数值修改为了transform_1
  2. 有一个自动放缩函数autoZoom,将<g>'transform'修改为了transform_2
  3. 用户再次进行拖动,<g>'transform'会从transform_1开始修改,因此会出现从transform_2transform_1的瞬移。

解决方案

得知原因之后,解决方案也非常明了。就是在任何需要进行放缩平移的地方,都将transform进行缓存,下一次再需要进行放缩平移操作时,从上一次的transform开始进行更改即可。

一开始我想的解决方案是在每次鼠标拖动时都记录一个偏移量,但是这个偏移量比较难获取,心想d3这么大个库应该不至于用这么蠢的办法,应该有更好用的方案。

查了一下官方的API,发现了一个叫zoomTransform(node)的接口,这个接口传入的是一个HTML node,需要用d3.select(xx).node()来获得,可以获取这个node的放缩数据。官方文档是这么说的:

Internally, an element’s transform is stored as element.__zoom; however, you should use this method rather than accessing it directly. If the given node has no defined transform, returns the transform of the closest ancestor, or if none exists, the identity transformation


在内部,元素的变换存储为 element.__zoom;但是,您应该使用此方法(指的是zoomTransform)而不是直接访问它。如果给定节点没有定义的变换,则返回最近祖先的变换,或者如果不存在,则返回恒等变换。返回的变换表示以下形式的二维变换矩阵(略):

These properties should be considered read-only; instead of mutating a transform, use transform.scale and transform.translate to derive a new transform.


这些属性应被视为只读;使用transform.scale和transform.translate来派生新的变换,而不是改变变换。(下文将介绍如何派生新的变换)

进一步查看了源码,发现在svg.call(zoom)这个操作后,<svg>这个HTML node就会绑上一个__zoom 属性,这个__zoom属性记录的是transform参数,也就是我们对<svg>进行的放缩平移变换。为此我还特定打印了一下,发现确实如此:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-YmbofakK-1689904577005)(/Users/zqqcee/Library/Application Support/typora-user-images/image-20230720100136154.png)]

那现在事情就变得很简单了,可以转变一下思路。之前我一直希望能够在autoZoom()之后,获得"zoom"事件的偏移量,使得我能够接着这个'transform'值修改。那么既然我无法获得偏移量,可以尝试在autoZoom()方法中不要直接修改<g>'transform'属性,而去修改<svg>.__zoom值。

放缩平移写法

在一开始时,使用d3.zoom()创建放缩对象zoom,并在任何时刻都使用<svg>call(zoom)修改放缩值。在绑定"zoom"事件时,因为<svg> callzoom,因此任何偏移量都会记录在<svg>,在修改<g>'transform'属性时,可以直接使用d3.zoomTransform(svg.node())来获得<svg>.__zoom来进行应用。

const svg = d3.select('body').append('svg');
const g = svg.append('g');
const zoom = d3.zoom().on('zoom',()=>{g.attr('transform', d3.zoomTransform(svg.node()));
})

特殊修改'transform'函数的写法

这里需要说明一下autoZoom()的写法,假设我们现在已经计算出了'transform'数值transformXtransformYk。现在需要修改<svg>__zoom属性为当前的'transform'数值。

查阅了官方文档,找到了可以使用的API:

  • d3.zoomIdentity。这个API可以创建一个新的'transform':{x:0,y:0,k:1},并允许使用transform.translate(x,y), transform.scale(k)对其进行更改。
  • selection.call(zoom.transform,new_transform);使用这个接口能够将<svg>.__zoom修改为new_transform

综上,代码为:

const new_transform = d3.zoomIdentity.translate(transformX, transformY).scale(k);
d3.select('svg').call(zoom.transform,new_transform);

总结

简而言之,任何对<g>的放缩与平移操作,都需要作用在<svg>上,并且使用<svg>.__zoom()来修改。

完整代码:

//zoom事件绑定
const svg = d3.select('body').append('svg');
const g = svg.append('g');
const zoom = d3.zoom().on('zoom',()=>{g.attr('transform', d3.zoomTransform(svg.node()));
})//需要修改特定transform的函数,以autoZoom为例
const autoZoom = (transformX,transformY,k) =>{const new_transform = d3.zoomIdentity.translate(transformX, transformY).scale(k);d3.select('svg').call(zoom.transform,new_transform);
}
http://www.shuangfujiaoyu.com/news/35092.html

相关文章:

  • 公司支付网站建设费进什么费用抖音seo关键词优化怎么做
  • 企业做网站平台的好处seo网络贸易网站推广
  • 免费制作软件的网站推广普通话黑板报
  • 泉州做网站便宜搜狗站长管理平台
  • 重庆百度网站快速排名seo研究中心教程
  • 编辑网站绑定 主机名最近的国内新闻
  • wordpress英文仿站教程搜狗竞价推广效果怎么样
  • 网站管理系统哪个好免费网站收录入口
  • 做网站一定需要主机吗企业模板建站
  • 海宁市住房与城乡规划建设局网站发布软文的平台
  • 微信订单网站模版网上推广渠道有哪些
  • wordpress login.php长春seo主管
  • 网站安全保障机制如何做怎么建立自己的网页
  • 做水电到哪个网站找信息爱站seo工具包官网
  • 做网站第一次见客户seo思维
  • 前端和网站部署做网站的搜索引擎关键词优化方案
  • 无极限网站模板百度快照seo
  • 深圳网站定制深圳网站建设公司哪家好淘宝网站的推广与优化
  • 小语种网站建设及推广seo网络营销推广
  • 新疆生产建设兵团安监局网站网站快速排名优化
  • 梅州建站规划教育培训网页设计
  • wordpress简洁百度免费优化
  • 网站建设 中企动力南通高端营销型网站
  • 网站建设 数据归属站长工具收录查询
  • wordpress备案信息企业seo排名优化
  • 做网站备负责人风险大吗永久免费二级域名申请
  • 注册送38元的游戏网站英文seo兼职
  • 个人免费域名注册网站怎样建立一个网络销售平台
  • 新网站怎么快速收录品牌推广计划书怎么写
  • 柳州企业 商家应该如何做网站厦门seo网络优化公司