@tensorflow-models/handposeを使ってWebで手を検出する
React
デモ https://fukekazki.github.io/react-handpose/
リポジトリ https://github.com/FukeKazki/react-handpose/tree/main
パッケージのインストール
以下のコマンドでtensorflow関連のパッケージを導入します。
pnpm add @tensorflow-models/handpose @tensorflow/tfjs-backend-webgl
メディアの権限
Webカメラにアクセスするためにはユーザーの承認が必要です。
const useVideo = (videoRef: RefObject<HTMLVideoElement>) => { const [isAllowed, setIsAllowed] = useState(false); useEffect(() => { const setupCamera = async () => { if (!videoRef.current) return; if (navigator.mediaDevices.getUserMedia) { const stream = await navigator.mediaDevices.getUserMedia({ video: true, }); videoRef.current.srcObject = stream; } }; setupCamera().then(() => { setIsAllowed(true); }); }, [videoRef]); return { isAllowed, } };
手の検出
最初に@tensorflow/tfjs-backend-webgl
の実行が必要です。
loadHanspose
はモデルの読み込みをしています。
detect
はモデルを使って0.1sごとに予測をし、予測結果をcanvasに描画しています。
import * as handpose from "@tensorflow-models/handpose"; import "@tensorflow/tfjs-backend-webgl"; const useHandpose = ( videoRef: RefObject<HTMLVideoElement>, canvasRef: RefObject<HTMLCanvasElement>, ) => { const [isLoading, setIsLoading] = useState(false); const [model, setModel] = useState<handpose.HandPose>(); useEffect(() => { const loadHandpose = async () => { const net = await handpose.load(); setModel(net); }; setIsLoading(true); loadHandpose().finally(() => { setIsLoading(false) }); }, []); useEffect(() => { const detect = async () => { if (!model) return; if (!videoRef.current) return; const predictions = await model.estimateHands(videoRef.current, true); if (!canvasRef.current) return; const ctx = canvasRef.current.getContext("2d"); if (!ctx) return; ctx.clearRect(0, 0, 640, 480); for (let i = 0; i < predictions.length; i++) { const keypoints = predictions[i].landmarks; for (let j = 0; j < keypoints.length; j++) { const [x, y] = keypoints[j]; ctx.beginPath(); ctx.arc(x, y, 5, 0, 3 * Math.PI); ctx.fillStyle = "aqua"; ctx.fill(); } } }; const interval = setInterval(detect, 100); // 0.1秒ごとに検出 return () => clearInterval(interval); }, [model, videoRef, canvasRef]); return { isLoading, } };
実行
上記のhookをAppコンポーネントから呼び出します。 UI部分はvideo domの上にcanvasを重ねて表示してます。
function App() { const videoRef = useRef<HTMLVideoElement>(null); const canvasRef = useRef<HTMLCanvasElement>(null); const { isAllowed } = useVideo(videoRef); const { isLoading } = useHandpose(videoRef, canvasRef); return ( <> {isAllowed || <p>Camera Permission Denied</p>} {isLoading && <p>Model Loading...</p>} <div style={{ position: "relative", width: 640, height: 480, }} > <video style={{ position: "absolute", top: 0, left: 0, transform: "scaleX(-1)", }} ref={videoRef} width="640" height="480" autoPlay /> <canvas style={{ position: "absolute", top: 0, left: 0, }} ref={canvasRef} width="640" height="480" /> </div> </> ); }
まとめ
手の検出ができました。👏 問題点として片手のみしか検出できていないため、両手を対応したい場合は他の方法(?)を試す必要がありそうです。
目次