PoseMaker / static /poseEditor.js
jonigata's picture
add translate/rotate/scale
61dfb1b
raw
history blame
6.63 kB
console.log("hello from poseEditor.js")
var canvas = null;
var ctx = null;
// candidateの形式:[[x1, y1, score1], [x2, y2, score2], ...]
const candidateSource = [
[235, 158, 0.93167633],
[234, 220, 0.97106987],
[193, 222, 0.93366587],
[138, 263, 0.87655306],
[89, 308, 0.8868227],
[276, 220, 0.9038924],
[325, 264, 0.92930061],
[375, 309, 0.9217211],
[207, 347, 0.86410147],
[203, 433, 0.86538297],
[199, 523, 0.95236528],
[261, 347, 0.88489777],
[262, 430, 0.90848708],
[261, 522, 0.94749999],
[227, 148, 0.94189668],
[245, 148, 0.93967074],
[208, 158, 0.92053992],
[258, 154, 0.73533273]
];
const candidate = candidateSource.map(point => [point[0], point[1] - 70, point[2]]);
// subsetの形式:[[index1, index2, ..., -1], [index1, index2, ..., -1], ...]
const subset = [
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 33.81122635, 18]
];
function clearCanvas() {
var w = canvas.width;
var h = canvas.height;
ctx.fillStyle = 'black';
ctx.fillRect(0, 0, w, h);
}
function resizeCanvas(width, height) {
canvas.width = width ? width : canvas.width;
canvas.height = height ? height : canvas.height;
clearCanvas();
drawBodyPose(candidate, subset);
}
function drawBodyPose(candidate, subset) {
const stickwidth = 4;
const limbSeq = [[2, 3], [2, 6], [3, 4], [4, 5], [6, 7], [7, 8], [2, 9], [9, 10],
[10, 11], [2, 12], [12, 13], [13, 14], [2, 1], [1, 15], [15, 17],
[1, 16], [16, 18], [3, 17], [6, 18]];
const colors = [[255, 0, 0], [255, 85, 0], [255, 170, 0], [255, 255, 0], [170, 255, 0], [85, 255, 0], [0, 255, 0],
[0, 255, 85], [0, 255, 170], [0, 255, 255], [0, 170, 255], [0, 85, 255], [0, 0, 255], [85, 0, 255],
[170, 0, 255], [255, 0, 255], [255, 0, 170], [255, 0, 85]];
for (let i = 0; i < 17; i++) {
for (let n = 0; n < subset.length; n++) {
const index0 = subset[n][limbSeq[i][0]-1];
const index1 = subset[n][limbSeq[i][1]-1];
if (index0 === -1 || index1 === -1) {
continue;
}
const [X0, Y0] = candidate[index0].slice(0, 2);
const [X1, Y1] = candidate[index1].slice(0, 2);
ctx.beginPath();
ctx.lineWidth=stickwidth;
ctx.strokeStyle = `rgb(${colors[i].join(',')})`;
ctx.moveTo(X0, Y0);
ctx.lineTo(X1, Y1);
ctx.stroke();
}
}
ctx.font = '12px serif';
for (let i = 0; i < 18; i++) {
for (let n = 0; n < subset.length; n++) {
const index = subset[n][i];
if (index === -1) {
continue;
}
const [x, y] = candidate[index].slice(0, 2);
ctx.beginPath();
ctx.arc(x, y, 4, 0, 2 * Math.PI);
ctx.fillStyle = `rgb(${colors[i].join(',')})`;
ctx.fill();
ctx.fillStyle = 'rgb(255,255,255)'
// ctx.fillText(index, x-3, y+4);
}
}
}
function getNearestCandidateIndex(x, y) {
let minDist = Infinity;
let minIndex = -1;
for (let i = 0; i < candidate.length; i++) {
const dist = Math.sqrt((x - candidate[i][0]) ** 2 + (y - candidate[i][1]) ** 2);
if (dist < minDist) {
minDist = dist;
minIndex = i;
}
}
if (16 < minDist) {
return -1;
}
return minIndex;
}
// ドラッグ中に座標を保持するための変数
let isDragging = false;
let dragIndex = -1;
let dragStartX = 0;
let dragStartY = 0;
let draggingCandidate = null;
function getCanvasPosition(event) {
const rect = canvas.getBoundingClientRect();
const x = event.clientX - rect.left;
const y = event.clientY - rect.top;
return [x, y];
}
// Canvas要素上でマウスが押された場合に呼び出される関数
function handleMouseDown(event) {
const [x, y] = getCanvasPosition(event);
const index = getNearestCandidateIndex(x, y);
// ドラッグ処理の開始
if (0 <= index || event.altKey || event.ctrlKey || event.shiftKey) {
isDragging = true;
dragIndex = index;
dragStartX = x;
dragStartY = y;
draggingCandidate = JSON.parse(JSON.stringify(candidate))
}
}
// Canvas要素上でマウスが動いた場合に呼び出される関数
function handleMouseMove(event) {
if (!isDragging) {
return;
}
const [x, y] = getCanvasPosition(event);
const dragOffsetX = x - dragStartX;
const dragOffsetY = y - dragStartY;
if (event.ctrlKey) {
let xScale = 1 + dragOffsetX / canvas.width;
let yScale = 1 + dragOffsetY / canvas.height;
for (let i = 0 ; i < candidate.length; i++) {
candidate[i][0] = (draggingCandidate[i][0] - dragStartX) * xScale + dragStartX;
candidate[i][1] = (draggingCandidate[i][1] - dragStartY) * yScale + dragStartY;
}
} else if (event.shiftKey) {
// rotate
let angle = Math.atan2(dragOffsetY, dragOffsetX);
for (let i = 0 ; i < candidate.length; i++) {
let x = draggingCandidate[i][0] - dragStartX;
let y = draggingCandidate[i][1] - dragStartY;
candidate[i][0] = x * Math.cos(angle) - y * Math.sin(angle) + dragStartX;
candidate[i][1] = x * Math.sin(angle) + y * Math.cos(angle) + dragStartY;
}
} else if (event.altKey) {
// 全体移動
for (let i = 0 ; i < candidate.length; i++) {
candidate[i][0] = draggingCandidate[i][0] + dragOffsetX;
candidate[i][1] = draggingCandidate[i][1] + dragOffsetY;
}
} else {
// 個別移動
candidate[dragIndex][0] = draggingCandidate[dragIndex][0] + dragOffsetX;
candidate[dragIndex][1] = draggingCandidate[dragIndex][1] + dragOffsetY;
}
clearCanvas();
drawBodyPose(candidate, subset);
}
// Canvas要素上でマウスが離された場合に呼び出される関数
function handleMouseUp(event) {
isDragging = false;
}
function initializePose(w,h) {
canvas = document.getElementById('canvas');
ctx = canvas.getContext('2d');
canvas.addEventListener('mousedown', handleMouseDown);
canvas.addEventListener('mousemove', handleMouseMove);
canvas.addEventListener('mouseup', handleMouseUp);
resizeCanvas(w, h);
}
function savePose() {
const canvasUrl = canvas.toDataURL();
const createEl = document.createElement('a');
createEl.href = canvasUrl;
// This is the name of our downloaded file
createEl.download = "pose.png";
createEl.click();
createEl.remove();
}