image-label-ts
Version:
基于ts的前端图片标注组件,不依赖vue和react
818 lines (810 loc) • 33 kB
JavaScript
import { SVG } from '@svgdotjs/svg.js';
import '@svgdotjs/svg.panzoom.js';
import '@svgdotjs/svg.draggable.js';
import '@svgdotjs/svg.topoly.js';
import { merge } from 'lodash-es';
var LayerTypeEnum;
(function (LayerTypeEnum) {
LayerTypeEnum["polygon"] = "polygon";
// rect = 'rect',
})(LayerTypeEnum || (LayerTypeEnum = {}));
class Layer {
containerCanvas;
type;
dom;
name;
id;
constructor() { }
}
class LayerCanvas {
constructor() { }
}
var img = "";
class SelectLayer {
containerCanvas;
layer;
selectDom;
cloneDom;
constructor(layer) {
this.layer = layer;
this.containerCanvas = layer.containerCanvas;
this.removeSelect();
this.selectDom = this.containerCanvas.canvas.group().size().attr({ id: 'select' });
this.cloneDom = this.layer.dom
.clone()
.attr({
stroke: this.layer.style.color,
fill: this.layer.style.selectFillColor,
name: 'select-polygon',
'stroke-dasharray': '10,10',
})
.on('click', function (e) {
e.stopPropagation();
})
.addTo(this.containerCanvas.canvas.findOne('#select'));
if (this.layer.able.remove) {
this.createCloseBtn();
}
if (this.layer.able.drag) {
this.selectDom.draggable().on('dragstart', () => {
// 拖动遮罩区域时,隐藏选中图形
layer.dom.hide();
});
this.createLayerDraggableCircle();
this.selectDom.on('dragend', this.layerMaskDragendHandler.bind(this));
}
}
static init(layer) {
return new SelectLayer(layer);
}
removeSelect() {
this.containerCanvas.canvas.find('#select').remove();
}
removeLayer() {
this.removeSelect();
this.layer.dom.remove();
this.containerCanvas.event.remove && this.containerCanvas.event.remove(this.layer);
}
createCloseBtn() {
let minIndex = 0;
let left = 5;
let top = -40;
const points = this.layer.dom.plot();
points.forEach((item, index) => {
if (item[1] < points[minIndex][1]) {
minIndex = index;
}
});
if (points[minIndex][0] + 45 > this.containerCanvas.width) {
left = -45;
}
if (points[minIndex][1] - 40 < 0) {
top = 0;
}
if (this.selectDom.closeBtn) {
this.selectDom.closeBtn.remove();
}
this.selectDom.closeBtn = this.containerCanvas.canvas
.image(new URL(img, import.meta.url).href)
.size(40, 40)
.attr({
x: points[minIndex][0] + left,
y: points[minIndex][1] + top,
style: 'cursor: pointer',
})
.on('click', () => {
this.removeLayer();
this.containerCanvas.removeLayerByLayer(this.layer);
})
.addTo(this.containerCanvas.canvas.findOne('#select'));
}
createLayerDraggableCircle() {
const points = [...this.cloneDom.array()];
points.pop();
points.map(([cx, cy], index) => {
const circle = this.selectDom.circle();
return circle
.radius(this.layer.style.circleRadius / this.containerCanvas.root.zoomNum)
.attr({
cx,
cy,
fill: this.layer.style.color,
index,
})
.draggable()
.on('click', (e) => {
e.stopPropagation();
})
.on('dragstart', () => {
this.layer.dom.hide();
})
.on('dragmove', this.layerMaskCircleDragmoveHandler.bind(this, circle))
.on('dragend', this.layerMaskDragendCircleHandler.bind(this));
});
}
layerMaskCircleDragmoveHandler(circle) {
const { cx, cy, index } = circle.attr(['cx', 'cy', 'index']);
const pointsArray = this.cloneDom.array();
pointsArray[index] = [cx, cy];
if (index === 0) {
pointsArray[pointsArray.length - 1] = [cx, cy];
}
this.cloneDom.plot(pointsArray.join().replace(/,/g, ' '));
}
layerMaskDragendHandler() {
this.layer.dom._array = this.cloneDom.array();
this.layer.dom.plot(this.cloneDom.array().join().replace(/,/g, ' ')).show();
this.dragEnd();
}
dragEnd() {
// 多边形
if (this.layer.type === LayerTypeEnum.polygon) {
const points = Polygon.transformSVGPointsToOrigin({
points: this.cloneDom.array(),
scale: this.containerCanvas.transform.scale,
offsetX: this.containerCanvas.transform.offsetX,
offsetY: this.containerCanvas.transform.offsetY,
});
this.layer.setOption({
points: points,
});
}
if (this.containerCanvas.event.dragEnd) {
this.containerCanvas.event.dragEnd(this.layer);
}
}
layerMaskDragendCircleHandler() {
this.layer.dom.plot(this.cloneDom.array().join().replace(/,/g, ' ')).show();
this.selectDom.closeBtn.remove();
this.selectDom.closeBtn = null;
this.createCloseBtn();
this.dragEnd();
}
}
const defaultIPolygonOption = {
name: '',
points: [],
style: {
color: '#E1600A',
fillColor: 'rgba(0, 0, 0, 0)',
selectFillColor: 'rgba(0, 0, 0, 0)',
strokeWidth: 2,
circleRadius: 4,
},
able: {
click: false,
drag: true,
remove: true,
},
};
class Polygon extends Layer {
points;
style;
able;
canvasOnce;
constructor(canvas, option) {
super();
this.containerCanvas = canvas;
const { x, y, width, height } = this.containerCanvas.labelImage.attr([
'x',
'y',
'width',
'height',
]);
this.canvasOnce = this.containerCanvas.canvas
.group()
.attr({ id: 'polygonCanvas-polygon-once', width, height });
this.canvasOnce.rect(width, height).attr({ x, y });
this.setOption(option);
if (!this.id) {
this.id = Math.floor(Math.random() * 1e10).toString();
}
this.draw();
}
static init(canvas, option) {
return new Polygon(canvas, option);
}
static transformOriginPointsToSVG(params) {
const { points, scale, offsetX, offsetY } = params;
return points.map((item) => [item.x / scale + offsetX, item.y / scale + offsetY]);
}
static transformSVGPointsToOrigin(params) {
const { points, scale, offsetX, offsetY } = params;
const res = points.map((item) => ({
x: Math.round((item[0] - offsetX) * scale),
y: Math.round((item[1] - offsetY) * scale),
}));
res.pop();
return res;
}
transformOriginPointsToSVG(points) {
const { scale = 1, offsetX = 0, offsetY = 0 } = this.containerCanvas.transform;
return points.map((item) => [item.x / scale + offsetX, item.y / scale + offsetY]);
}
transformSVGPointsToOrigin(points) {
const { scale = 1, offsetX = 0, offsetY = 0 } = this.containerCanvas.transform;
const res = points.map((item) => ({
x: Math.round((item[0] - offsetX) * scale),
y: Math.round((item[1] - offsetY) * scale),
}));
res.pop();
return res;
}
remove() {
this.dom.remove();
}
select() {
this.containerCanvas.event.select(this);
this.containerCanvas.selectLayer = SelectLayer.init(this);
}
// todo 此api出错率高,可做单元测试
setOption(option) {
this.type = LayerTypeEnum.polygon;
const _option = merge({}, defaultIPolygonOption, this, JSON.parse(JSON.stringify(option))); // 深度合并
Object.keys(_option).forEach((key) => {
this[key] = option[key] ? _option[key] : this[key] || defaultIPolygonOption[key];
});
}
draw() {
const pointsFormat = this.transformOriginPointsToSVG(this.points).map((item) => {
return ['L', item[0], item[1]];
});
pointsFormat.unshift([
'M',
pointsFormat[pointsFormat.length - 1][1],
pointsFormat[pointsFormat.length - 1][2],
]);
const polyPath = this.containerCanvas.canvas
.path()
.plot(['M', pointsFormat[0][1], pointsFormat[0][2]].join().replace(/,/g, ' '))
.attr({
stroke: this.style.color,
fill: 'none',
'stroke-width': this.style.strokeWidth / this.containerCanvas.root.zoomNum,
});
try {
this.dom = polyPath
.plot(pointsFormat.join().replace(/,/g, ' '))
.toPoly()
.attr({
color: this.style.color,
stroke: this.style.color,
'stroke-width': this.style.strokeWidth / this.containerCanvas.root.zoomNum,
fill: this.style.fillColor,
type: 'polygon',
})
.addTo(this.containerCanvas.canvas);
}
catch (e) {
console.error('draw error', e);
this.canvasOnce.remove();
this.canvasOnce = null;
}
this.dom.attr({ name: this.name });
if (this.able.click) {
this.dom.on('click', (e) => {
this.select();
e.stopPropagation();
});
}
this.canvasOnce.remove();
this.canvasOnce = null;
return this;
}
// todo
reDraw(option) {
this.setOption(option);
// ...
}
}
class PolygonCanvas extends LayerCanvas {
containerCanvas;
polygonCanvas;
tempPath; // 描述手动绘图的dom对象
polyPath; // 描述手动绘图的dom对象
closePoint; // 手动绘图时首尾连通的那个点
constructor(canvas) {
super();
this.containerCanvas = canvas;
const { x, y, width, height } = this.containerCanvas.labelImage.attr([
'x',
'y',
'width',
'height',
]);
this.polygonCanvas = this.containerCanvas.canvas
.group()
.attr({ id: 'drawLayer', width, height });
this.polygonCanvas
.rect(width, height)
.attr({ x, y, 'fill-opacity': 0 })
.on('mousemove', this.mouseMoveFollow.bind(this));
this.polygonCanvas.pathPoints = [];
this.tempPath = null;
this.polygonCanvas.on('mousemove', this.drawTempPath.bind(this));
this.polygonCanvas.on('mousemove', this.polygonCountIsClose.bind(this));
this.polygonCanvas.on('click', this.drawPolyPathHandler.bind(this));
}
static init(canvas) {
return new PolygonCanvas(canvas);
}
drawTempPath(e) {
if (this.polygonCanvas.pathPoints.length > 0) {
const [x, y] = this.currentNodeMovePosition(e);
const tempPathArray = this.polyPath.array();
tempPathArray[this.polygonCanvas.pathPoints.length] = ['L', x, y];
if (!this.tempPath) {
this.tempPath = this.polygonCanvas
.path()
.plot(tempPathArray.join().replace(/,/g, ' '))
.attr({
stroke: this.containerCanvas.style.color,
fill: this.containerCanvas.style.fillColor,
'fill-opacity': 0.6,
'stroke-width': this.containerCanvas.style.strokeWidth / this.containerCanvas.root.zoomNum,
'stroke-dasharray': '10,10',
});
}
else {
this.tempPath.plot(tempPathArray.join().replace(/,/g, ' '));
}
}
}
// 点击回调事件,绘点
drawPolyPathHandler(e) {
const [x, y] = this.currentNodeMovePosition(e);
if (this.polygonCanvas.pathPoints.length === 0) {
this.polygonCanvas.pathPoints.push(this.polygonCanvas
.circle()
.radius(this.containerCanvas.style.circleRadius / this.containerCanvas.root.zoomNum)
.attr({ cx: x, cy: y, fill: this.containerCanvas.style.color, id: 'begin' }));
this.polyPath = this.polygonCanvas
.path()
.plot(['M', x, y].join().replace(/,/g, ' '))
.attr({
stroke: this.containerCanvas.style.color,
fill: 'none',
'stroke-width': this.containerCanvas.style.strokeWidth / this.containerCanvas.root.zoomNum,
});
}
else {
this.polygonCanvas.pathPoints.push(this.polygonCanvas
.circle()
.radius(this.containerCanvas.style.circleRadius / this.containerCanvas.root.zoomNum)
.attr({ cx: x, cy: y, fill: this.containerCanvas.style.color }));
// pathArray 为构成图形的点位
const pathArray = this.polyPath.array();
pathArray.push(['L', x, y]);
this.polyPath.plot(pathArray.join().replace(/,/g, ' '));
}
}
// 检测鼠标是否移动到了多边形的第一个点(是否形成封闭多边形)
polygonCountIsClose(e) {
if (this.polygonCanvas.pathPoints.length > 2) {
const [x, y] = this.currentNodeMovePosition(e);
const { cx, cy } = this.polygonCanvas.pathPoints[0].attr(['cx', 'cy']);
const a = Math.abs(cx - x);
const b = Math.abs(cy - y);
if (Math.sqrt(a * a + b * b) < 15 / this.containerCanvas.root.zoomNum) {
if (!this.closePoint) {
this.closePoint = this.polygonCanvas
.circle()
.radius(15 / this.containerCanvas.root.zoomNum)
.attr({
cx,
cy,
fill: this.containerCanvas.style.color,
'fill-opacity': 0.5,
})
.on('click', (e) => {
this.pathToPolygon();
e.stopPropagation();
});
}
this.polygonCanvas.followCircle && this.polygonCanvas.followCircle.hide();
this.closePoint?.show();
const tempPathArray = this.polyPath.array();
tempPathArray[this.polygonCanvas.pathPoints.length] = ['L', cx, cy];
this.tempPath.plot(tempPathArray.join().replace(/,/g, ' '));
}
else {
if (this.closePoint) {
this.closePoint.remove();
this.closePoint = null;
}
this.polygonCanvas.followCircle && this.polygonCanvas.followCircle.show();
}
}
}
pathToPolygon() {
const points = this.polyPath.array().map((item) => {
return [item[1], item[2]];
});
const formatPoints = Polygon.transformSVGPointsToOrigin({
points,
scale: this.containerCanvas.transform.scale,
offsetX: this.containerCanvas.transform.offsetX,
offsetY: this.containerCanvas.transform.offsetY,
});
this.containerCanvas.drawLayer(LayerTypeEnum.polygon, {
name: this.containerCanvas.layerName,
points: formatPoints,
style: {
color: this.containerCanvas.style.color,
fillColor: this.containerCanvas.style.fillColor,
selectFillColor: this.containerCanvas.style.selectFillColor,
strokeWidth: this.containerCanvas.style.strokeWidth,
circleRadius: this.containerCanvas.style.circleRadius,
},
able: {
click: true,
drag: true,
remove: true,
},
});
this.remove();
this.containerCanvas.drawDone && this.containerCanvas.drawDone();
}
remove() {
if (this.containerCanvas.canvas.findOne('#drawLayer')) {
this.containerCanvas.canvas.findOne('#drawLayer').remove();
}
this.polygonCanvas = null;
this.tempPath = null;
this.polyPath = null;
this.closePoint = null;
}
currentNodeMovePosition(e) {
const bgNode = this.containerCanvas.root.findOne('#labelImage-background').node;
const { x, y } = bgNode === null ? { x: 0, y: 0 } : bgNode.getClientRects()[0];
return [
(1 / this.containerCanvas.root.zoomNum) *
(e.clientX - x + this.containerCanvas.transform.offsetX),
(1 / this.containerCanvas.root.zoomNum) *
(e.clientY - y + this.containerCanvas.transform.offsetY),
];
}
mouseMoveFollow(e) {
const [cx, cy] = this.currentNodeMovePosition(e);
this.containerCanvas.root.on('mousemove', this.removeFollower.bind(this));
const polygonCanvas = this.polygonCanvas;
if (!polygonCanvas.followCircle) {
polygonCanvas.followCircle = polygonCanvas
.circle()
.radius(this.containerCanvas.style.circleRadius / this.containerCanvas.root.zoomNum)
.attr({ cx, cy, fill: this.containerCanvas.style.color })
.on('mousemove', (e) => {
const [x, y] = this.currentNodeMovePosition(e);
polygonCanvas.followCircle.attr({ cx: x, cy: y });
});
}
else {
polygonCanvas.followCircle.attr({ cx, cy });
}
}
removeFollower(e) {
if (this.containerCanvas.root) {
const polygonCanvas = this.polygonCanvas;
const [cx, cy] = this.currentNodeMovePosition(e);
const { width, height } = polygonCanvas.attr(['width', 'height']);
if (cx > width || cy > height || cx < 0 || cy < 0) {
if (polygonCanvas.followCircle) {
polygonCanvas.followCircle.remove();
polygonCanvas.followCircle = null;
this.containerCanvas.root.off('mousemove');
}
}
}
}
}
const loadImage = (url) => {
return new Promise((res) => {
const image = new Image();
image.onload = () => {
res(image);
};
image.src = url;
});
};
var ModeEnum;
(function (ModeEnum) {
ModeEnum["default"] = "default";
ModeEnum["drag"] = "drag";
ModeEnum["poly"] = "poly";
})(ModeEnum || (ModeEnum = {}));
const defaultContainerCanvasOption = {
width: 0,
height: 0,
zoom: {
zoomFactor: 0.5,
zoomMin: 0.5,
zoomMax: 20,
},
style: {
color: '#E1600A',
fillColor: 'rgba(0, 0, 0, 0)',
selectFillColor: 'rgba(0, 0, 0, 0)',
strokeWidth: 2,
circleRadius: 4,
},
layerName: 'default-layer',
mode: ModeEnum.default,
event: {
draw: () => { },
select: () => { },
remove: () => { },
dragEnd: () => { },
clickCanvas: () => { },
beforeLoadImage: () => { },
loadedImage: () => { },
},
transform: {
scale: 1,
fixedOffset: false,
offsetX: 0,
offsetY: 0,
},
};
class ContainerCanvas {
root;
canvas;
layerCanvas; // layer画布, 手动画图时使用
layers = []; // 标注的图形
labelImage; // 标注的图
selectLayer;
polygonCanvas;
width;
height;
zoom;
style;
layerName;
mode;
event;
transform;
imageUrl = '';
drawImageWidth = 0;
drawImageHeight = 0;
constructor(el, option) {
this.setOption(option);
this.root = SVG().addTo(el).attr({ style: 'vertical-align: bottom' });
this.root.zoomNum = 1;
this.canvas = this.root.group().attr({ id: 'canvas' });
this.root.size(this.width, this.height).viewbox(0, 0, this.width, this.height);
if (this.mode !== ModeEnum.default) {
this.root = this.root.panZoom(this.zoom);
}
this.root.on('zoom', (lvl) => {
this.root.zoomNum = lvl.detail.level;
this.canvas.find('path').forEach((path) => {
path.attr({
'stroke-width': this.style.strokeWidth / lvl.detail.level,
});
});
this.canvas.find('polyline').forEach((polyline) => {
polyline.attr({
'stroke-width': this.style.strokeWidth / lvl.detail.level,
});
});
this.canvas.find('circle').forEach((circle) => {
circle.radius(this.style.circleRadius / lvl.detail.level);
});
});
this.canvas.on('click', () => {
this.selectLayer && this.selectLayer.removeSelect();
this.selectLayer = undefined;
this.event.clickCanvas && this.event.clickCanvas();
});
}
static init(el, option) {
return new ContainerCanvas(el, option);
}
setOption(option) {
const _option = merge({}, defaultContainerCanvasOption, option);
Object.keys(_option).forEach((key) => {
// this[key] = _option[key];
this[key] = this[key] || _option[key];
});
}
setMode(mode) {
this.mode = mode;
switch (mode) {
case ModeEnum.drag:
this.polygonCanvas && this.polygonCanvas.remove();
this.polygonCanvas = null;
break;
case ModeEnum.poly:
// 创建layer画布
this.polygonCanvas = PolygonCanvas.init(this);
break;
case ModeEnum.default:
break;
}
}
toggleMode(mode1, mode2) {
if (this.mode === mode1) {
this.setMode(mode2);
}
else {
this.setMode(mode1);
}
}
// 清除整个画布
clear() {
this.canvas.children().forEach((child) => {
child.remove();
});
this.layers = [];
}
// 获取所有的图形 (ContainerCanvas实例是个ref时, containerCanvas.layers也会变成响应式)
getLayers(type) {
return this.layers.filter((layer) => layer.type === type);
}
// 获取所有的图形
getLayersByName(type, name) {
return this.getLayers(type).filter((layer) => layer.name === name);
}
// 根据多边形删除多边形
removeLayerByLayer(layer) {
layer.remove();
for (let i = 0; i < this.layers.length; i++) {
if (this.layers[i] === layer) {
this.layers.splice(i, 1);
return;
}
}
}
// 根据多边形删除多边形
removeLayerById(id) {
const layer = this.layers.find(item => item.id === id);
if (layer) {
layer.remove();
const index = this.layers.indexOf(layer);
this.layers.splice(index, 1);
}
}
drawDone() {
this.root.off('mousemove');
this.setMode(ModeEnum.drag);
this.event.draw && this.event.draw(this.layers[this.layers.length - 1]);
}
loadImageStack = [];
loadImagePool = (imageUrl, fn) => {
return new Promise((resolve) => {
const p = loadImage(imageUrl);
this.loadImageStack.push(p);
p.then((image) => {
if (this.loadImageStack[this.loadImageStack.length - 1] === p) {
fn.call(this, image);
}
this.loadImageStack.splice(0, this.loadImageStack.indexOf(p));
resolve(null);
});
});
};
// 加载图片
async loadImage(imageUrl, imageWidth, imageHeight) {
this.imageUrl = imageUrl;
if (this.mode !== ModeEnum.default) {
this.setMode(ModeEnum.drag);
}
this.root.zoom(1);
this.root.zoomNum = 1;
if (this.mode !== ModeEnum.default) {
this.root = this.root.panZoom(this.zoom);
}
this.root.viewbox(0, 0, this.width, this.height);
this.clear();
this.layers = [];
if (!imageUrl) {
return;
}
this.event.beforeLoadImage && this.event.beforeLoadImage();
await this.loadImagePool(imageUrl, (image) => {
this.event.loadedImage && this.event.loadedImage();
imageWidth = imageWidth || image.width;
imageHeight = imageHeight || image.height;
this.transform.scale = Math.max(imageWidth / this.width, imageHeight / this.height);
this.drawImageWidth = imageWidth / this.transform.scale;
this.drawImageHeight = imageHeight / this.transform.scale;
if (!this.transform.fixedOffset) {
this.transform.offsetX = (this.width - this.drawImageWidth) / 2;
this.transform.offsetY = (this.height - this.drawImageHeight) / 2;
}
this._loadImage(imageUrl).size(this.drawImageWidth, this.drawImageHeight).attr({
x: this.transform.offsetX,
y: this.transform.offsetY,
});
});
}
_loadImage(imageUrl) {
const image = this.canvas.findOne('#labelImage-background');
if (image) {
image.load(imageUrl);
return image;
}
else {
this.labelImage = this.canvas.image(imageUrl).attr({ id: 'labelImage-background' });
return this.labelImage;
}
}
// 画蒙层
drawMask(color) {
this.canvas
.rect(this.drawImageWidth, this.drawImageHeight)
.fill(color)
.attr({ x: this.transform.offsetX, y: this.transform.offsetY });
}
// 画一个图片的多边形可使区域
drawPolygonImages(points, imageUrl = this.imageUrl) {
if (!points || !points.length)
return;
const pattern = this.root
.defs()
.pattern()
.attr({ x: this.transform.offsetX, y: this.transform.offsetY });
pattern.image(imageUrl).size(this.drawImageWidth, this.drawImageHeight);
points.forEach(point => {
const _points = Polygon.transformOriginPointsToSVG({
points: point,
scale: this.transform.scale,
offsetX: this.transform.offsetX,
offsetY: this.transform.offsetY,
});
const str = _points.map((item) => `${item[0]},${item[1]}`).join(' ');
this.canvas.polygon(str).fill(pattern);
});
}
// 绘制一个图形
drawLayer(type, option) {
if (type === LayerTypeEnum.polygon) {
try {
this.layers.push(Polygon.init(this, option));
}
catch (e) {
console.error('drawLayer error', e);
}
}
}
// 绘制多个图形
drawLayers(type, options) {
options.forEach((option) => {
option.points.forEach((point) => {
this.drawLayer(type, {
...option,
points: point,
});
});
});
}
hideLayers(type) {
this.getLayers(type).forEach((item) => {
item.dom.style.display = 'none';
});
}
showLayers(type) {
this.getLayers(type).forEach((item) => {
item.dom.style.display = 'block';
});
}
hideLayersByName(type, name) {
this.getLayersByName(type, name).forEach((item) => {
item.dom.node.style.display = 'none';
});
}
showLayersByName(type, name) {
this.getLayersByName(type, name).forEach((item) => {
item.dom.node.style.display = 'block';
});
}
// 切换图片并且绘制多边形
async loadImageAndDrawLayers(imageUrl, options) {
await this.loadImage(imageUrl);
(options || []).forEach((option) => {
if (option.type === LayerTypeEnum.polygon) {
this.drawLayers(option.type, option.option);
}
});
}
}
export { ContainerCanvas, Layer, LayerCanvas, LayerTypeEnum, ModeEnum, Polygon, PolygonCanvas, SelectLayer, defaultContainerCanvasOption };
//# sourceMappingURL=index.js.map