data:image/s3,"s3://crabby-images/a32b6/a32b680c30d27535f36b16f9dcc5008eb94b9601" alt=""
data:image/s3,"s3://crabby-images/00ed9/00ed9e1fbb24ed79e1f7b57b61e076aee33bb054" alt=""
・SVG芸人
・イラストレーター
・複雑GUI会
・Vue.js Japan UG
・deno_ja
hashrock
data:image/s3,"s3://crabby-images/e468f/e468f15c25a7c7b99b85ab86541e44c15e5356dc" alt=""
data:image/s3,"s3://crabby-images/36446/364461fbda50524e0e4b916eff3eabd125e0c538" alt=""
SVG芸人が
よく聞かれる質問
「何でSVG?」
- なんで数ある技術の中からSVGなの?
- SVGの何が嬉しいの?
- っていうかSVG芸人って何?
即答できない
- 一言で説明しづらい
- 一切使わなくてもフロントエンドは作れる
- 使ってるうちに楽しくなってくる系
「使って楽しい」が
伝わればいいな…
お品書き
- 平成のワードアート
- 未来のワードアート
- ワードアートのSVG実装
- ついでにエディタの実装
ワードアートとは
data:image/s3,"s3://crabby-images/89f80/89f8083d3a5decadbcaf6243985e91b6ef4324ff" alt=""
data:image/s3,"s3://crabby-images/0eb86/0eb860faa4317966e1feffa26755c69999bbfe53" alt=""
data:image/s3,"s3://crabby-images/2fca8/2fca80ed9f2ef162ceb17a2b12012f5f192c31c8" alt=""
ワードアート
- テキストをデザインできるMS Officeの共通モジュール
- かなり昔のOfficeからある
- 確認した中で最古のWordArtは1994年
- https://makewordart.com/
data:image/s3,"s3://crabby-images/89f80/89f8083d3a5decadbcaf6243985e91b6ef4324ff" alt=""
特徴的な極太書体
波打ってる
data:image/s3,"s3://crabby-images/2fca8/2fca80ed9f2ef162ceb17a2b12012f5f192c31c8" alt=""
激しいグラデーション
data:image/s3,"s3://crabby-images/0eb86/0eb860faa4317966e1feffa26755c69999bbfe53" alt=""
ドロップシャドウ
3D風変形エフェクト
data:image/s3,"s3://crabby-images/80c96/80c96f4f7ce8c46645c8b6894744893200660acd" alt=""
MS WordでのWordArtの
つくりかた
data:image/s3,"s3://crabby-images/8670d/8670da572d2fe3d0981f2a75380397089e96e195" alt=""
data:image/s3,"s3://crabby-images/3bb23/3bb230de0dd94e015fca73d5957ef2a2b3720da4" alt=""
data:image/s3,"s3://crabby-images/b8336/b833678a1f0b6560cd3fefdafd697d331d628cbc" alt=""
1994年(平成6年)
当時のWordArt
令和のワードアート
SVGならどこまでできる?
SVGで作ってみました
踊る
data:image/s3,"s3://crabby-images/8463b/8463b6f9210c02790f04b26fc640658f773d8aae" alt=""
回る
data:image/s3,"s3://crabby-images/69cd2/69cd260d34da6567108c3b53d0e5780cc59e3b4d" alt=""
セレクタブル
data:image/s3,"s3://crabby-images/c16f3/c16f3742aa1c87fc03debf32654ba1a1c6275d53" alt=""
contenteditable
data:image/s3,"s3://crabby-images/c59ae/c59ae017626be9bfc79612bee31c34ebe421c2e3" alt=""
SVGすごい!
っていうかブラウザがすごい
SVGでのワードアートの作り方
<text x="40" y="80">Hello World</text>
data:image/s3,"s3://crabby-images/b55a7/b55a700c8049d4d5574ad67351e92c0f36713990" alt=""
1.テキスト要素
data:image/s3,"s3://crabby-images/e828a/e828a189519207bdc0db4084aca5892e4f235593" alt=""
<text x="40" y="80">Hello World</text>
<path
fill="none"
stroke="blue"
stroke-width="3"
d="M 28 152 C 28 152 49 78 100 67 C 151 56 210 164 255 148 C 300 133 295 59 295 59"
/>
2.テキスト
曲線
<text x="40" y="80">
<textPath href="#MyPath">
Hello World
</textPath>
</text>
<path
id="MyPath"
fill="none"
stroke="blue"
stroke-width="3"
d="M 28 152 C 28 152 49 78 100 67 C 151 56 210 164 255 148 C 300 133 295 59 295 59"
/>
data:image/s3,"s3://crabby-images/71ad0/71ad09be4530946880eadc2a430db1f4571ca338" alt=""
3.テキスト
+曲線
<textPath>を追加
テキストと曲線を
関連付ける
<path
id="MyPath"
fill="none"
stroke="blue"
stroke-width="3"
d="M 28 152 C 28 152 49 78 100 67 C 151...."
/>
「d属性」が曲線を定義している
しかし手書きはきつい
リアルタイム編集できれば便利そう…
SVG DOM は DOM Level 2 と互換性を保つように構築されている。
(W3C SVG 1.1 仕様)
SVGもDOMから扱える
ReactやVueなどのFWはSVGに対応している
バインディングも当然可能
data:image/s3,"s3://crabby-images/6283b/6283ba2e381add3a64191c98da839bda03664b85" alt=""
Vueで曲線エディタを実装
3次ベジェ曲線
よく使うやつ
<path d="M70 110 C 70 140, 110 140, 110 110" />
data:image/s3,"s3://crabby-images/6a828/6a82843df9f3fac97afa5bd2a34c8ea35224305b" alt=""
<path d="M70 110 C 70 140, 110 140, 110 110" />
data:image/s3,"s3://crabby-images/6a828/6a82843df9f3fac97afa5bd2a34c8ea35224305b" alt=""
a
b
c
d
a
b
c
d
4つの制御点をもとに曲線を描画
data:image/s3,"s3://crabby-images/70bcf/70bcff8f5b3a1ccc3b8967d573dfbedc13875b18" alt=""
ベジェ曲線を手描きしてみよう
data:image/s3,"s3://crabby-images/74c11/74c11a278b6d34b724cea92e4b9b6b1c2603a182" alt=""
data:image/s3,"s3://crabby-images/9f3c4/9f3c4208f82c8d132553927074f2b4851862d90a" alt=""
data:image/s3,"s3://crabby-images/1c111/1c1111620da0374861477b47aea6e8455225d317" alt=""
data:image/s3,"s3://crabby-images/f2fa8/f2fa81e24f6a017b01669da2508a5535770ee636" alt=""
data:image/s3,"s3://crabby-images/3ae68/3ae681c48fd17c5713f57b53aebed5c81caf7c53" alt=""
data:image/s3,"s3://crabby-images/5eb34/5eb34207872df2130e19cc7db2970c8e8f25cef9" alt=""
<path :d="curvesStr" id="path1" />
points: [
{ x: 200, y: 200 },
{ x: 240, y: 200 },
{ x: 210, y: 250 },
{ x: 250, y: 250 }
]
curvesStr() {
return this.curves
.map(i => {
return `M ${i.points[0].x},${i.points[0].y} C ${i.points[1].x},${i.points[1].y} ${i.points[2].x},${i.points[2].y} ${i.points[3].x},${i.points[3].y}`;
})
.join(" ");
}
座標の配列(data内)
computedで文字列化してd属性に渡す
<path :d="curvesStr" id="path1" />
points: [
{ x: 200, y: 200 },
{ x: 240, y: 200 },
{ x: 210, y: 250 },
{ x: 250, y: 250 }
]
curvesStr() {
return this.curves
.map(i => {
return `M ${i.points[0].x},${i.points[0].y} C ${i.points[1].x},${i.points[1].y} ${i.points[2].x},${i.points[2].y} ${i.points[3].x},${i.points[3].y}`;
})
.join(" ");
}
この4つの点を
編集するUIを作ればOK
頂点・ハンドルの
ドラッグ
data:image/s3,"s3://crabby-images/17383/173838c0b0e9285e78d6721b01d5c6c81df797ec" alt=""
基本的にはHTML要素のドラッグと
変わらないはず
ただし注意点あり
data:image/s3,"s3://crabby-images/1792e/1792e625dd16fd457c87fd3fe95813079c78aa9c" alt=""
ユーザー単位の世界
(単位なし)
スクリーン単位の世界
(px)
data:image/s3,"s3://crabby-images/1792e/1792e625dd16fd457c87fd3fe95813079c78aa9c" alt=""
MouseEventから取れる座標は単位px
SVG側の単位に変換が必要
function screenToSvg(point, el, svg) {
const pt = svg.createSVGPoint();
pt.x = point.x;
pt.y = point.y;
return pt.matrixTransform(el.getScreenCTM().inverse());
}
ユーティリティ関数を作って対応
詳しくはviewBox / getScreenCTMなどの
キーワードで検索
data:image/s3,"s3://crabby-images/8dc3d/8dc3d38c4ab3ca746adcccaa1309db749123db53" alt=""
きちんと対応すれば変形がかかった要素内でも
正常にドラッグ可能
data:image/s3,"s3://crabby-images/eb24b/eb24b8d0dfb370756d1b6bc257aa4caddc58a733" alt=""
data:image/s3,"s3://crabby-images/b95c6/b95c6a73f0db30b26be37db480326b9e449dc1ca" alt=""
ついでに作ったカラーピッカー
派手にする
data:image/s3,"s3://crabby-images/ab055/ab05582d702b300dcc17cdb6f9f88aa3809689b0" alt=""
Drop Shadow
SVGフィルタ
- defs下に定義する
- 組み合わせるといろんなフィルタを作れる
- そのものズバリのDrop Shadowはない
<filter id="dropshadow">
<feGaussianBlur in="SourceAlpha" stdDeviation="1" />
<feOffset dx="5" dy="5" result="offsetblur" />
<feComponentTransfer>
<feFuncA type="linear" slope="0.5" />
</feComponentTransfer>
<feMerge>
<feMergeNode />
<feMergeNode in="SourceGraphic" />
</feMerge>
</filter>
data:image/s3,"s3://crabby-images/763ad/763adbccba042a646a78f0079be3785825c4a164" alt=""
data:image/s3,"s3://crabby-images/2b2ea/2b2eaf07409d6374190c885829bc6d9290547630" alt=""
グラデーション
data:image/s3,"s3://crabby-images/0630d/0630d4026165fde39ac6c8b522cd54a98aa12d99" alt=""
線形グラデーション
- (x1, y1), (x2, y2)の2点間をグラデーションする
- グラデーション定義もバインディングして変更できる
<linearGradient
gradientUnits="userSpaceOnUse"
id="rainbow"
:x1="gradient.start.x"
:y1="gradient.start.y"
:x2="gradient.end.x"
:y2="gradient.end.y"
>
<stop offset="0%" stop-color="#ff0000" />
<stop offset="16.7%" stop-color="#ffff00" />
<stop offset="33.3%" stop-color="#00ff00" />
<stop offset="50.0%" stop-color="#00ffff" />
<stop offset="66.7%" stop-color="#0000ff" />
<stop offset="83.3%" stop-color="#ff00ff" />
<stop offset="100%" stop-color="#ff0000" />
</linearGradient>
DEMO
できなかったこと
-
グラデーションエディタの実装
-
SVG出力(Inkscapeで読むと壊れます)
-
キーボード操作対応
-
複数選択
-
3D変形
どうしてSVGなのか?
svgをGUI構築フレームワークとして考えることも可能
UIコンポーネント構築のベースとして
使い勝手が良い
自由度が高く
hackyになりにくい
(テキスト周りは除く)
作ったものが
自分の想定を
超えてくる感がある
(個人の感想です)
作ったものの使いみちが特にないので…
data:image/s3,"s3://crabby-images/a8a8a/a8a8a6600cfecb915d545bffca5d92cb483050fe" alt=""
とりあえずTシャツにしました
SVGでワードアート
By hash rock
SVGでワードアート
- 9,470