2
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

d3.jsでobjectの上に格子状の枠を作成する

Last updated at Posted at 2020-09-19

目的

完成イメージのように、ステータスを表示するダッシュボード画面上に四角いオブジェクトがあり、格子状の枠を作って色を塗りつぶしたり、テキストを入れるのがゴールです。使っているのはJavaScriptのライブラリd3.jsです。

ポイントは2つ。

  • foreignObject1要素で、オブジェクト内部にtableを追加し、格子状の枠を作る。
  • clipPath2要素で、角を丸くする。

初めは座標位置を指定してpathやareaで力技で描画することを考えたのですが、テキストのセンタリングやオブジェクトの拡大・縮小で破綻するので止めて、Webで色々調べつつまとめてみました。

本記事は、d3.jsでコードを書いたことがある人を前提に書いています。

完成イメージ

image.png

完成版HTML

最初に完成形を貼っておきます。

grid.html
<html>
  <head>
    <script src="https://d3js.org/d3.v6.min.js"></script>
    <style>
/* おまじない */
table {
  border-collapse: collapse;
  border-spacing: 0;
}
    </style>
  </head>
  <body>
    <div id="svg-area"></div>
    <script>
'use strict';
let data = [
  {t:'RED',c:'#ff0000'},
  {t:'GREEN',c:'#00ff00'},
  {t:'BLUE',c:'#0000ff'}
];
let width = 1000;
let height = 500;
let gridSize = 300;
let gap = 8;
let svg = d3.select("#svg-area").append("svg")
  .attr("width",width)
  .attr("height",height);

/* 角Rをつけた四角い図形でくり抜く */
svg.append("clipPath")
  .attr("id","clip")
  .selectAll("rect")
  .data(data)
  .enter()
  .append("rect")
  .attr("width", gridSize-gap)
  .attr("height", gridSize)
  .attr("x", (d,i) => i*gridSize)
  .attr("y", 0)
  .attr("rx", 16)
  .attr("ry", 16);

let g = svg.append("g")
  .attr("width",width)
  .attr("height",height);

let cards = g.selectAll("rect").data(data);

/* ベースとなる四角 */
cards.enter().append("rect")
  .attr("id",(d,i) => "card-"+i)
  .attr("x", (d,i) => i*gridSize)
  .attr("y", 0)
  .attr("width", gridSize-gap)
  .attr("height", gridSize)
  .attr("clip-path", "url(#clip)")
  .style("fill", "#000000");

/* tableタグのためのforeignObject要素 */
let table = cards.enter().append("foreignObject")
  .attr("x", (d,i) => i*gridSize)
  .attr("y", 0)
  .attr("width", gridSize-gap)
  .attr("height",gridSize)
  .attr("clip-path", "url(#clip)")
  .append("xhtml:table")
  .attr("width", gridSize-gap)
  .attr("height",gridSize)
  .attr("border",1)
  .attr("frame","void")
  .attr("bordercolor","#ffffff");

/* 通常のtableタグと同様に表の内部を作る */
let tr1 = table.append("tr");
let tr2 = table.append("tr");
let tr3 = table.append("tr");
tr1.append("td")
  .attr("height","33%")
  .attr("width","50%")
  .attr("bgcolor", (d) => d.c);
tr1.append("td")
  .attr("height","33%")
  .attr("width","50%")
  .style("text-align", "center")
  .append("font")
  .attr("color", "#ffffff")
  .text((d) => d.t);

tr2.append("td")
  .attr("height","33%");

tr2.append("td")
  .attr("height","33%");

tr3.append("td")
  .attr("colspan",2);
    </script>
  </body>
</html>

ヘッダ

細かく見ていきます。

ヘッダはシンプルにd3.jsライブラリを読み込むだけです。なお、styleタグの記述がないと枠にスペースが出来てしまいます。3

<html>
  <head>
    <script src="https://d3js.org/d3.v6.min.js"></script>
    <style>
/* おまじない */
table {
  border-collapse: collapse;
  border-spacing: 0;
}
    </style>
  </head>

ないとこんな感じ。
image.png

ボディ

こちらもシンプルで、div要素のみ。

  <body>
    <div id="svg-area"></div>
    <script>
    ...
    </script>
  </body>

オブジェクトを作る

次に、オブジェクトを実際に作っていきます。まずはベースとなる四角いオブジェクトを作成します。

'use strict';
let data = [
  {t:'RED',c:'#ff0000'},
  {t:'GREEN',c:'#00ff00'},
  {t:'BLUE',c:'#0000ff'}
];
let width = 1000;
let height = 500;
let gridSize = 300;
let gap = 8;
let svg = d3.select("#svg-area").append("svg")
  .attr("width",width)
  .attr("height",height);

let g = svg.append("g")
  .attr("width",width)
  .attr("height",height);

let cards = g.selectAll("rect").data(data);

/* ベースとなる四角 */
cards.enter().append("rect")
  .attr("id",(d,i) => "card-"+i)
  .attr("x", (d,i) => i*gridSize)
  .attr("y", 0)
  .attr("width", gridSize-gap)
  .attr("height", gridSize)
//  .attr("clip-path", "url(#clip)")
  .style("fill", "#000000");

ここまで以下の図形となります。
image.png

格子状の枠を作る

内部にtableタグを作成することで枠を作ります。foreignObjectを使ってそれを重ねるイメージです。分かりやすいように以下のような2x2の表を作るとします。

<table>
  <tr>
    <td></td>
    <td></td>
  </tr>
  <tr>
    <td></td>
    <td></td>
  </tr>
</table>

以下のようなコードになります。

/* tableタグのためのforeignObject要素 */
let table = cards.enter().append("foreignObject")
  .attr("x", (d,i) => i*gridSize)
  .attr("y", 0)
  .attr("width", gridSize-gap)
  .attr("height",gridSize)
//  .attr("clip-path", "url(#clip)")
  .append("xhtml:table")
  .attr("width", gridSize-gap)
  .attr("height",gridSize)
  .attr("border",1)
  .attr("frame","void")
  .attr("bordercolor","#ffffff");

/* 完成版HTMLと違います */
let tr1 = table.append("tr");
let tr2 = table.append("tr");
tr1.append("td")
  .attr("height","50%")
  .attr("width","50%");
tr1.append("td")
  .attr("height","50%")
  .attr("width","50%");
tr2.append("td")
  .attr("height","50%")
  .attr("width","50%");
tr2.append("td")
  .attr("height","50%")
  .attr("width","50%");

するとこんな感じで格子状の枠が作れました。完成イメージにするには、3段にする、背景色を指定する、カラムを結合する、テキストを加えるなど、通常のtableと同じような作業をやります。
image.png

角を丸くする

最後に角を丸くするには、clipPathを使います。描画部分を窓枠のようにマスクするイメージで、Web検索すれば色々出てきます。

以下のコードを追加し、上記解説のclip-path属性のコメントアウトを外します。

/* 角Rをつけた四角い図形でくり抜く */
svg.append("clipPath")
  .attr("id","clip")
  .selectAll("rect")
  .data(data)
  .enter()
  .append("rect")
  .attr("width", gridSize-gap)
  .attr("height", gridSize)
  .attr("x", (d,i) => i*gridSize)
  .attr("y", 0)
  .attr("rx", 16)
  .attr("ry", 16);

こちらになります。
image.png

まとめ

いかがでしたでしょうか?完成イメージのような図形でも、自由度の高いd3.jsライブラリを使えば割と簡単に描くことが出来るのがいいところです。

  1. <foreignObject> - SVG: Scalable Vector Graphics | MDN (https://developer.mozilla.org/ja/docs/Web/SVG/Element/foreignObject)

  2. <clipPath> - SVG: Scalable Vector Graphics | MDN (https://developer.mozilla.org/ja/docs/Web/SVG/Element/clipPath)

  3. テーブルのセルの隙間をリセットするCSS (https://qiita.com/macer_fkm/items/bac56f0f863a19cfd674)

2
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
2
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?