ソースはこちらです
動作サンプルページ (github pages)
ポイント
-
canvasはDOMオブジェクトを直接操作する必要があるため
useRef
フックで利用する -
canvasの大きさはCSSではなく属性で設定する。
const canvas = useRef(null);
<canvas ref={canvas}
width={props.width} height={props.height}
onMouseDown={mouseDown}
~~~ 以下省略 ~~~
機能
-
手書きでお絵描きをするコンポーネントです。マウスドラッグ中に線を引きます。
-
propsで下記機能を利用できます
-
線の太さ、色を変更可能(lineHeigh, lineColor)
-
コンポーネントのサイズを指定する(width, height)
-
クリアプロパティ(clear: boolean)を変更すると領域をクリアします(親コンポーネントのから子コンポーネントの機能を呼び出す方法がこれしか思いつかなかった。functionコンポーネントでやり方あるんですかね?)
-
1本ごと線描画完了時に、onUpdateCanvas(canvas)をコールバックします。
⇒ 引数canvasでDOM自体を引き渡すため、画像をダウンロードするなど出来ます。
-
コンポーネントの仕組み
- 手書きコンポーネントソース全体
- ドラッグ中(mouseDown⇒mouseMove)に線を描画します
- マウスを離す(onMouseUp)か、canvas領域から出る(onMouseLeave)と線描画を終了します
- useEffectで
props.clear
の値変更を監視し、canvasをクリアします
import React, { useRef, useState, useEffect } from "react";
/**
* 手書きコンポーネントprops定義
*/
export type HandWritingAttribute = {
width?: number,
height?: number,
lineWidth?: number,
lineColor?: string,
lineCap?: CanvasLineCap,
clear?: boolean,
onUpdateCanvas?: (e: HTMLCanvasElement) => void,
}
/**
* 手書きコンポーネント(ドラッグ中線を描画する)
* ・props経由で、線の太さ、色を指定
* ・onUpdateCanvas()で線描画毎にコールバック
* ・props.clearの値をトグルする毎にクリア
* @param props
* @returns
*/
const HandWriting: React.FC<HandWritingAttribute> = (props) => {
// canvasはDOMを直接操作するためuseRef()経由で操作する
const canvas = useRef(null);
// ドラッグ中判断フラグ(マウスを離すか、canvas外へ出たらfalse)
const [drawing, setDrawing] = useState(false);
// 領域クリア用。親コンポーネントでclearの値を変更するとcanvasをクリアする(toggle時常にクリア)
useEffect(() => {
const ctx = (canvas.current as HTMLCanvasElement).getContext('2d');
if( ctx ) {
ctx.clearRect(0, 0, props.width, props.height);
if (props.onUpdateCanvas) props.onUpdateCanvas(canvas.current);
}
}, [props.clear]);
// 描画に必要なcontextを取得し、線の色、幅をセットする
const getContext = () => {
const ctx = (canvas.current as HTMLCanvasElement).getContext('2d');
ctx.lineWidth = props.lineWidth;
ctx.lineCap = props.lineCap;
ctx.strokeStyle = props.lineColor;
return ctx;
}
// 線描画開始処理。beginPath()で新しいパスを開始する(開始しないと色や太さが変更できない)
const mouseDown: React.MouseEventHandler = (e) => {
const { offsetX: x ,offsetY: y } = e.nativeEvent;
setDrawing(true);
const ctx = getContext();
ctx.beginPath();
ctx.moveTo(x, y);
}
// マウスの動きに合わせて線を描画する
const mouseMove: React.MouseEventHandler = (e) => {
if (!drawing) return;
const { offsetX: x ,offsetY: y } = e.nativeEvent;
const ctx = getContext();
ctx.lineTo(x, y);
ctx.stroke();
}
// 線描画完了(canvas更新イベントコールバックを行う)
const endDrawing = () => {
setDrawing(false);
if (props.onUpdateCanvas) props.onUpdateCanvas(canvas.current);
}
// canvas
return (
<>
<canvas ref={canvas}
width={props.width} height={props.height}
onMouseDown={mouseDown}
onMouseMove={mouseMove}
onMouseUp={endDrawing}
onMouseLeave={endDrawing} />
</>
);
};
// propsのデフォルト値を設定
HandWriting.defaultProps = {
width: 500,
height: 300,
lineWidth: 10,
lineColor: "rgb(100, 100, 100)",
lineCap: "round",
};
export default HandWriting;
利用方法
- 利用サンプルソース全体
import React from 'react';
import HandWriting from './HandWrinting';
function App() {
const [dataUrl, setDataUrl] = React.useState(null);
const [settings, setSettings] = React.useState({lineWidth:7, lineColor: '#5555bb', clear: false})
// 画像コンポーネント更新コールバック
const onUpdateCanvas = (e: HTMLCanvasElement) => {
// 画像をstateに保存し下記の用途で利用する
// ⇒<img>タグに表示
// ⇒画像のダウンロード
setDataUrl(e.toDataURL('image/png'));
}
// 画像ダウンロード
const downloadCanvasImage = () => {
const dlLink = document.createElement("a");
dlLink.href = dataUrl;
dlLink.download = 'handwriting.png';
dlLink.click();
dlLink.remove();
}
// 手書きコンポーネント線の幅変更
const setLineWidth = (e: React.ChangeEvent<HTMLSelectElement>) => {
setSettings( prev => {
return {...prev, lineWidth: parseInt(e.target.value, 10)}
});
}
// 手書きコンポーネント線の色変更
const setColor= (e: React.ChangeEvent<HTMLInputElement>) => {
setSettings( prev => {
return {...prev, lineColor: e.target.value}
});
}
// 手書きコンポーネントクリア
const clearComponent= () => {
setSettings( prev => {
return {...prev, clear: !prev.clear}
});
}
const lineWidth = [1,2,3,5,7,10,14,20];
return (
<>
<h1>HandWritingコンポーネント利用サンプル</h1>
<div>
<h2>HandWritingコンポーネント</h2>
<div>
<label htmlFor="lineWidth">線の太さ:</label>
<select name="lineWidth" value={settings.lineWidth} onChange={setLineWidth}>
{lineWidth.map((i) =><option key={i} value={i}>{i}</option>)}
</select>
<label htmlFor="lineColor">色:</label>
<input name="lineColor" type="color" value={settings.lineColor} onChange={setColor}></input>
<button onClick={clearComponent}>クリア</button>
</div>
<HandWriting onUpdateCanvas={onUpdateCanvas} {...settings} />
</div>
<div>
<h2>HandWritingの更新イベントサンプル(画像更新時にimgタグに反映)</h2>
<img id="newImg" alt="" src={dataUrl} />
</div>
<div>
<button type="button" onClick={downloadCanvasImage}>
HandWriting画像のダウンロード
</button>
</div>
</>
);
}
export default App;
画像をimgタグに反映する
-
Canvasの更新コールバックで、画像に変換し<img>タグに表示します
-
dataUrlは画像をbase64エンコーディングした文字列がセットされます。
~
// 画像コンポーネント更新コールバック
const onUpdateCanvas = (e: HTMLCanvasElement) => {
// 画像をstateに保存し下記の用途で利用する
// ⇒<img>タグに表示
// ⇒画像のダウンロード
setDataUrl(e.toDataURL('image/png'));
}
<img id="newImg" alt="" src={dataUrl} />
画像のダウンロード
- dataUrlにセットした画像データをダウンロードするため<a>を動的に生成します。
const downloadCanvasImage = () => {
const dlLink = document.createElement("a");
dlLink.href = src;
dlLink.download = 'handwriting.png';
dlLink.click();
dlLink.remove();
}
参考にしたサイト
classコンポーネントをfunctionに書き換え、イベント処理を追加しました。