PaperJS,需要 select 使用鼠标按下的透明光栅下的项目

PaperJS, Need to select items underneath a transparent raster using Mouse Down

我有一个 Canvas 有多个光栅图像。 我在 Tool 上使用 onMouseDown 来查找 select 被点击的项目。

我有一个新要求。

假设,两张图片互相重叠,上面的图片是部分透明的。这使得较低的图像可见。但是当我尝试点击下方图片时,显然我最终选择了上方图片。

尝试失败

我尝试在 Raster 上使用 getPixel(point) 函数。我想如果我能确定 selected 像素是透明的,我就可以忽略该光栅并寻找其他项目。但是我没有使用此功能获得我期望的颜色值(透明或不透明)。

所以,我的第二个想法是我需要将 mousedown 事件点从全局坐标 space 更改为局部光栅坐标 space。还是不行。

有没有办法实现我想要的?

代码

tool.onMouseDown = (event) => {    
      project.activeLayer.children.forEach((item) => {

    if (item.contains(event.point)) {
          // check if hit was on a transparent raster pixel
          const pixel = item.getPixel(event.point)
          console.error(pixel.toCSS(true))

         // 2nd attempt
          const pixel = item.getPixel(item.globalToLocal(event.point))
          console.error(pixel.toCSS(true))
        }
    }
}

有一种更简单的方法可以实现您想要实现的目标。
您可以依靠 project.hitTestAll() 方法对所有项目进行命中测试。
那么,如果命中项是一个光栅,命中像素的颜色信息将包含在hitResult.color中。 hitResult.color.alpha 是检查光栅是否命中非透明像素所需的全部内容。

这是解决方案的 sketch 演示。

const dataUrl = '';
const lowOpacity = 0.3;

// create 2 rasters
new Raster({
    source: dataUrl,
    opacity: lowOpacity,
    onLoad: function() {
        this.position = view.center - 100;
    }
});
new Raster({
    source: dataUrl,
    opacity: lowOpacity,
    onLoad: function() {
        this.position = view.center + 100;
    }
});

// on mouse down
function onMouseDown(event) {
    // unselect previously selected items
    paper.project.selectedItems.forEach(item => {
        item.selected = false;
        item.opacity = lowOpacity;
    });

    // do a hit test on all project items
    const hitResults = project.hitTestAll(event.point);
    // for each hit result
    for (let i = 0; i < hitResults.length; i++) {
        const hitResult = hitResults[i];
        // if item was hit on a non transparent pixel
        if (hitResult && hitResult.color && hitResult.color.alpha > 0) {
            // select item
            hitResult.item.selected = true;
            hitResult.item.opacity = 1;
            // break loop
            break;
        }
    }
}