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;
}
}
}
我有一个 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;
}
}
}