fabricjs中因mac retina屏幕像素比导致的捕捉canvas像素不准确

背景

问题的起因是某项目中实现一个光标指向canvas后,捕捉当前鼠标位置色值的功能。功能完成后部分用户出现色值偏差,如图:

分析

fabricjs初始化canvas时,enableRetinaScalling默认开启,致使canvas getImageData()坐标错位,

问题解决:

修改前:

javascript 复制代码
/**
 * 获取canvas背景图中指定像素色值
 * @param canvas canvas参数为new fabric.Canvas()实例
 * @param x 鼠标x坐标位置
 * @param y 鼠标y坐标位置
 * @returns 当前鼠标所在像素的色值
 */
function getColorAtPosition(canvas, x, y) {
  var canvasElement = canvas.getElement()
  var context = canvasElement.getContext('2d')
  let newX = x * canvas.backgroundImage.scaleX
  let newY = y * canvas.backgroundImage.scaleY
  var pixel = context.getImageData(newX, newY, 1, 1).data
  const color = new fabric.Color(`rgba(${pixel.join(',')})`)
  return {
    array: pixel,
    rgba: `rgba(${pixel.join(',')})`,
    hex: `#${color.toHex()}`
  }
}

为保证当前坐标在不同屏幕下都可以正常捕捉,需要乘以屏幕像素比的比例,window.devicePixelRatio

修改后:

javascript 复制代码
// 获取canvas背景图中指定像素色值
function getColorAtPosition(canvas, x, y) {
  var canvasElement = canvas.getElement()
  var context = canvasElement.getContext('2d')
  // 注:retina屏幕下需要乘以当前设备的devicePixelRatio值 
  let newX = x * canvas.backgroundImage.scaleX * window.devicePixelRatio
  let newY = y * canvas.backgroundImage.scaleY * window.devicePixelRatio
  var pixel = context.getImageData(newX, newY, 1, 1).data
  const color = new fabric.Color(`rgba(${pixel.join(',')})`)
  return {
    array: pixel,
    rgba: `rgba(${pixel.join(',')})`,
    hex: `#${color.toHex()}`
  }
}

可以根据不同屏幕像素比,计算在canvas图像中相应的准确坐标。

相关推荐
Laravel技术社区35 分钟前
用PHP8实现斗地主游戏,实现三带一,三带二,四带二,顺子,王炸功能(第二集)
前端·游戏·php
m0_738120722 小时前
应急响应——知攻善防Web-3靶机详细教程
服务器·前端·网络·安全·web安全·php
hh随便起个名8 小时前
力扣二叉树的三种遍历
javascript·数据结构·算法·leetcode
我是小路路呀8 小时前
element级联选择器:已选中一个二级节点,随后又点击了一个一级节点(仅浏览,未确认选择),此时下拉框失去焦点并关闭
javascript·vue.js·elementui
程序员爱钓鱼8 小时前
Node.js 编程实战:文件读写操作
前端·后端·node.js
PineappleCoder9 小时前
工程化必备!SVG 雪碧图的最佳实践:ID 引用 + 缓存友好,无需手动算坐标
前端·性能优化
JIngJaneIL9 小时前
基于springboot + vue古城景区管理系统(源码+数据库+文档)
java·开发语言·前端·数据库·vue.js·spring boot·后端
敲敲了个代码9 小时前
隐式类型转换:哈基米 == 猫 ? true :false
开发语言·前端·javascript·学习·面试·web
澄江静如练_9 小时前
列表渲染(v-for)
前端·javascript·vue.js