Skip to content

Commit 7b717c1

Browse files
authored
Merge pull request #15 from DDD-Community/feat#14
[Feat#14] 스냅샷 가이드라인 기능 추가
2 parents 894e9dd + a543cc2 commit 7b717c1

File tree

5 files changed

+365
-9
lines changed

5 files changed

+365
-9
lines changed

src/components/PoseDetector.tsx

+64-6
Original file line numberDiff line numberDiff line change
@@ -13,14 +13,19 @@ const PoseDetector: React.FC = () => {
1313
const [isTextNeck, setIsTextNeck] = useState<boolean | null>(null)
1414
const [isModelLoaded, setIsModelLoaded] = useState<boolean>(false)
1515
const [mode, setMode] = useState<string>("snapshot")
16-
16+
const [canInit, setCanInit] = useState<boolean>(false)
17+
const [isSnapSaved, setIsSnapSaved] = useState<boolean>(false);
1718
const modelRef = useRef<any>(null)
1819
const snapRef = useRef<pose[] | null>(null)
1920
const resultRef = useRef<pose[] | null>(null)
2021
const textNeckStartTime = useRef<number | null>(null)
2122
const timer = useRef<any>(null)
2223
const canvasRef = useRef<HTMLCanvasElement>(null)
2324

25+
const dx = useRef<number>(0)
26+
const dy = useRef<number>(0)
27+
const scale = useRef<number>(1)
28+
2429
const { requestNotificationPermission, showNotification } = usePushNotification()
2530

2631
const requestApi = (delay: number): Promise<void> => new Promise((resolve) => setTimeout(resolve, delay))
@@ -64,10 +69,23 @@ const PoseDetector: React.FC = () => {
6469
await window.ml5.setBackend("webgl")
6570
}
6671

72+
const canInitCallback = (canInit : boolean) => {
73+
setCanInit(canInit)
74+
}
75+
6776
const detect = useCallback(
6877
(results: pose[]): void => {
6978
resultRef.current = results
70-
if (canvasRef.current) drawPose(results, canvasRef.current)
79+
if (canvasRef.current) {
80+
drawPose(
81+
results,
82+
canvasRef.current,
83+
dx.current,
84+
dy.current,
85+
scale.current,
86+
canInitCallback,
87+
!!snapRef.current)
88+
}
7189
if (snapRef.current) {
7290
const _slope = detectSlope(snapRef.current, results, mode === "snapshot")
7391
const _isTextNeck = detectTextNeck(snapRef.current, results, mode === "snapshot")
@@ -108,7 +126,10 @@ const PoseDetector: React.FC = () => {
108126
)
109127

110128
const getInitSnap = (): void => {
111-
if (modelRef && modelRef.current) snapRef.current = resultRef.current
129+
if (modelRef && modelRef.current) {
130+
snapRef.current = resultRef.current
131+
setIsSnapSaved(true)
132+
}
112133
}
113134

114135
useEffect(() => {
@@ -129,6 +150,8 @@ const PoseDetector: React.FC = () => {
129150
setIsTextNeck(null)
130151
setSlope(null)
131152
snapRef.current = null
153+
setIsSnapSaved(false)
154+
setCanInit(false)
132155
}
133156

134157
const onChangeMode = (e: React.ChangeEvent<HTMLSelectElement>) => {
@@ -138,6 +161,25 @@ const PoseDetector: React.FC = () => {
138161
}
139162
}
140163

164+
const onChangeTranslation = (e: React.ChangeEvent<HTMLInputElement>) => {
165+
const id = e.target.id ;
166+
if(e.target.value){
167+
const value = Number.parseInt(e.target.value)
168+
switch(id){
169+
case 'vertical' :
170+
dy.current = value
171+
return
172+
case 'horizontal' :
173+
dx.current = value
174+
return
175+
case 'scale' :
176+
scale.current = value / 100 * 2
177+
return
178+
default :
179+
}
180+
}
181+
}
182+
141183
const onCancelAutoPoseMonitoring = () => {
142184
initializePoseMonitoring()
143185
}
@@ -164,6 +206,14 @@ const PoseDetector: React.FC = () => {
164206
/>
165207
{isModelLoaded && (
166208
<>
209+
<div>
210+
<div>좌우 이동</div>
211+
<input id="horizontal" type="range" min={-100} max={100} onChange={onChangeTranslation}></input>
212+
<div>상하 이동</div>
213+
<input id="vertical" type="range" min={-100} max={100} onChange={onChangeTranslation}></input>
214+
<div>크기 변경</div>
215+
<input id="scale" type="range" min={0} max={100} onChange={onChangeTranslation}></input>
216+
</div>
167217
<div className="font-bold text-red-500">본 화면은 좌우가 반대로 보이고 있으니 주의하세요!</div>
168218
<div>
169219
<select className="rounded border border-gray-400 bg-white p-2" onChange={onChangeMode}>
@@ -177,9 +227,17 @@ const PoseDetector: React.FC = () => {
177227
스냅샷 모드입니다. 올바른 자세를 하신 후에, 버튼을 눌러 촬영을 하면 해당 자세를 기준으로 부적절한
178228
자세를 추적합니다!
179229
</div>
180-
<button className="rounded bg-blue-500 px-4 py-2 font-bold text-white" onClick={getInitSnap}>
181-
올바른 자세를 촬영한 후 자세 측정 시작!
182-
</button>
230+
{
231+
canInit ?
232+
isSnapSaved ?
233+
<button className="rounded bg-blue-500 px-4 py-2 font-bold text-white" onClick={initializePoseMonitoring}>
234+
스냅샷 다시 찍기
235+
</button> :
236+
<button className="rounded bg-blue-500 px-4 py-2 font-bold text-white" onClick={getInitSnap}>
237+
올바른 자세를 촬영한 후 자세 측정 시작!
238+
</button> :
239+
<div className="font-bold text-red-500">스냅샷을 찍을 수 없습니다. 가이드 라인에 맞게 자세를 잡아주세요.</div>
240+
}
183241
</>
184242
)}
185243
{mode === "skeleton" && (

src/utils/calculator.ts

+67
Original file line numberDiff line numberDiff line change
@@ -126,3 +126,70 @@ export const getMidPoint = (p1: point, p2: point): point => {
126126

127127
return { x, y };
128128
};
129+
130+
/**
131+
* x=axisX를 기준으로 point를 대칭 이동 시킴
132+
*
133+
* @param {point} point
134+
* @param {number} axisX
135+
*
136+
*/
137+
export const getSymmetricPointFromX = (point : point, axisX : number) => {
138+
return {
139+
x: axisX + (axisX - point.x),
140+
y: point.y
141+
};
142+
}
143+
144+
/**
145+
* y=axisY를 기준으로 point를 대칭 이동 시킴
146+
*
147+
* @param {point} point
148+
* @param {number} axisY
149+
*
150+
*/
151+
export const getSymmetricPointFromY = (point : point, axisY : number) => {
152+
return {
153+
x: point.x,
154+
y: axisY + (axisY -point.y)
155+
};
156+
}
157+
158+
/**
159+
* dx, dy 만큼 point를 이동 시킴
160+
*
161+
* @param {point} point
162+
* @param {number} dx
163+
* @param {number} dy
164+
*
165+
*/
166+
export const getTranslatedPoint = (point : point, dx : number, dy : number) => {
167+
return {
168+
x: point.x + dx,
169+
y: point.y + dy
170+
}
171+
}
172+
173+
/**
174+
* cx, cy를 기준으로 scaleFactor 만큼 point의 스케일을 변경
175+
*
176+
* @param {point} point
177+
* @param {number} cx
178+
* @param {number} cy
179+
* @param {number} scaleFactor
180+
*
181+
*/
182+
export const getScaledPoint = (point : point, cx : number, cy : number, scaleFactor : number) => {
183+
const x = point.x
184+
const y = point.y
185+
186+
// 기준점으로부터의 상대적인 거리 계산
187+
const dx = x - cx
188+
const dy = y - cy
189+
190+
// 스케일 팩터를 적용한 새로운 거리 계산
191+
const scaledX = cx + dx * scaleFactor
192+
const scaledY = cy + dy * scaleFactor
193+
194+
return { x: scaledX, y: scaledY }
195+
}

src/utils/drawer.ts

+39-3
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,51 @@
11
import type { pose } from "@/utils/detector"
2+
import { getSymmetricPointFromY, getTranslatedPoint, getScaledPoint } from "./calculator";
3+
import { guideLinePoints } from "./guideLine";
4+
5+
export const drawPose = (poses: pose[], canvas: HTMLCanvasElement, dx : number = 0, dy : number = 0, scale : number = 1, canInitCallback : (canInit : boolean) => void, isInit : boolean) : void => {
26

3-
export const drawPose = (poses: pose[], canvas: HTMLCanvasElement): void => {
47
const ctx = canvas.getContext("2d")
8+
59
if (ctx) {
610
ctx.clearRect(0, 0, canvas.width, canvas.height)
711

12+
const guideLine = new Path2D()
13+
const origin = {x : canvas.width / 2, y : canvas.height}
14+
15+
// canavs 좌표계 기준으로 가이드 라인 정렬 (가이드 라인의 기준은 (canvas.width / 2, canvas.height))
16+
let _guideLinePoints = guideLinePoints.map(point => getSymmetricPointFromY(point,canvas.height / 2.))
17+
if(dx!==0 || dy!==0) _guideLinePoints = _guideLinePoints.map(point => getTranslatedPoint(point,dx,dy))
18+
if(scale!==1) _guideLinePoints = _guideLinePoints.map(point=>getScaledPoint(point, origin.x, origin.y, scale))
19+
guideLine.moveTo(_guideLinePoints[0].x, _guideLinePoints[0].y)
20+
for (let i = 1; i < _guideLinePoints.length; i++) {
21+
guideLine.lineTo(_guideLinePoints[i].x, _guideLinePoints[i].y)
22+
}
23+
guideLine.closePath()
24+
ctx.strokeStyle = 'blue'
25+
ctx.fillStyle = 'rgba(0, 0, 255, 0.3)';
26+
ctx.stroke(guideLine)
27+
ctx.fill(guideLine)
28+
829
poses.forEach((pose) => {
9-
// 왼쪽과 오른쪽 어깨 이어주는 선 그리기
1030
const leftShoulder = pose.keypoints.find((kp) => kp.name === "left_shoulder")
1131
const rightShoulder = pose.keypoints.find((kp) => kp.name === "right_shoulder")
32+
const leftEar = pose.keypoints.find((kp) => kp.name === "left_ear")
33+
const rightEar = pose.keypoints.find((kp) => kp.name === "right_ear")
1234

35+
//
36+
if(leftShoulder && rightShoulder && leftEar && rightEar){
37+
if(!isInit){
38+
if(
39+
ctx.isPointInPath(guideLine,leftShoulder?.x, leftShoulder?.y) &&
40+
ctx.isPointInPath(guideLine,rightShoulder?.x, rightShoulder?.y) &&
41+
ctx.isPointInPath(guideLine,leftEar?.x, leftEar?.y) &&
42+
ctx.isPointInPath(guideLine,rightEar?.x, rightEar?.y)
43+
) canInitCallback(true)
44+
else canInitCallback(false)
45+
}
46+
}
47+
48+
// 왼쪽과 오른쪽 어깨 이어주는 선 그리기
1349
if (leftShoulder && rightShoulder && leftShoulder.confidence > 0.2 && rightShoulder.confidence > 0.2) {
1450
ctx.beginPath()
1551
ctx.moveTo(leftShoulder.x, leftShoulder.y)
@@ -29,4 +65,4 @@ export const drawPose = (poses: pose[], canvas: HTMLCanvasElement): void => {
2965
})
3066
})
3167
}
32-
}
68+
}

0 commit comments

Comments
 (0)