Skip to content

SVGで簡易ビジュアライザ

概要

  • Scalable Vector Graphics
  • xmlベースの2次元ベクタ形式イメージ用画像形式

表示方法

  • chromeなどで開く
  • macでpngに変換
    • qlmanage -t -s 1024 -o . output.svgoutput.svg.pngに変換できる
  • jupyter notebook
    • from IPython.display import SVG, displayした上で、display(SVG(...))で出力できる

簡易利用クラス

struct RGB {
    int r, g, b;

    RGB() : r(0), g(0), b(0) {
    }
    RGB(int r, int g, int b) : r(r), g(g), b(b) {
    }
    // 0〜1の値の色付け
    void set(double val) {
        val = min(val, 1.0);
        val = max(val, 0.0);
        if (val < 0.5) {
            double x = val * 2.0;
            r = (int)(30.0 * (1.0 - x) + 144.0 * x);
            g = (int)(144.0 * (1.0 - x) + 255 * x);
            b = (int)(255.0 * (1.0 - x) + 30.0 * x);
        } else {
            double x = val * 2.0 - 1.0;
            r = (int)(144.0 * (1.0 - x) + 255 * x);
            g = (int)(255.0 * (1.0 - x) + 30.0 * x);
            b = (int)(30.0 * (1.0 - x) + 70.0 * x);
        }
    }
    string to_string() const {
        stringstream ss;
        ss << "#";
        ss << std::hex << std::setw(2) << std::setfill('0') << r;
        ss << std::hex << std::setw(2) << std::setfill('0') << g;
        ss << std::hex << std::setw(2) << std::setfill('0') << b;
        return ss.str();
    }
};

class SimpleSVG {
    int H, W;
    vector<string> xmls;

    void add_arrow_marker() {
        stringstream ss;
        ss << "<marker id=\"arrow\" viewBox=\"-5 -5 10 10\" orient=\"auto\">";
        ss << " <polygon points=\"-5,-5 5,0 -5,5\" fill=\"black\" stroke=\"none\"/>";
        ss << "</marker>";
        xmls.push_back(ss.str());
    }

    void init() {
        add_arrow_marker();
    }

   public:
    SimpleSVG(int H, int W) : H(H), W(W) {
        init();
    }

    void rect(int y, int x, int h, int w, const string& fill = "none",
              const string& stroke = "black", const string& stroke_width = "1") {
        stringstream ss;
        ss << "<rect";
        ss << " x=\"" << x << "\" y=\"" << y << "\"";
        ss << " width=\"" << w << "\" height=\"" << h << "\"";
        ss << " fill=\"" << fill << "\"";
        ss << " stroke=\"" << stroke << "\"";
        ss << " stroke-width=\"" << stroke_width << "\"";
        ss << " />";
        xmls.push_back(ss.str());
    }

    void rect_with_wall(int y, int x, int h, int w, bool U = false, bool D = false, bool L = false,
                        bool R = false, const string& fill = "none", const string& stroke = "black",
                        const string& stroke_width = "1", const string& wall_stroke_width = "5") {
        rect(y, x, h, w, fill, stroke, stroke_width);
        if (U) line(y, x, y, x + w, stroke, wall_stroke_width);
        if (D) line(y + h, x, y + h, x + w, stroke, wall_stroke_width);
        if (L) line(y, x, y + h, x, stroke, wall_stroke_width);
        if (R) line(y, x + w, y + h, x + w, stroke, wall_stroke_width);
    }

    void circle(int cy, int cx, int r, const string& fill = "none", const string& stroke = "black",
                const string& stroke_width = "1") {
        stringstream ss;
        ss << "<circle";
        ss << " cx=\"" << cx << "\" cy=\"" << cy << "\"";
        ss << " r=\"" << r << "\"";
        ss << " fill=\"" << fill << "\"";
        ss << " stroke=\"" << stroke << "\"";
        ss << " stroke-width=\"" << stroke_width << "\"";
        ss << " />";
        xmls.push_back(ss.str());
    }

    void line(int y1, int x1, int y2, int x2, const string& stroke = "black",
              const string& stroke_width = "1") {
        stringstream ss;
        ss << "<line";
        ss << " x1=\"" << x1 << "\" y1=\"" << y1 << "\"";
        ss << " x2=\"" << x2 << "\" y2=\"" << y2 << "\"";
        ss << " stroke=\"" << stroke << "\"";
        ss << " stroke-width=\"" << stroke_width << "\"";
        ss << " />";
        xmls.push_back(ss.str());
    }

    void arrow(int y1, int x1, int y2, int x2, const string& stroke_width = "4") {
        stringstream ss;
        ss << "<line";
        ss << " x1=\"" << x1 << "\" y1=\"" << y1 << "\"";
        ss << " x2=\"" << x2 << "\" y2=\"" << y2 << "\"";
        ss << " stroke=\"black\"";
        ss << " stroke-width=\"" << stroke_width << "\"";
        ss << " marker-end=\"url(#arrow)\"";
        ss << " />";
        xmls.push_back(ss.str());
    }

    // 左上揃え: dominant_baseline=hanging, text_anchor=start
    void text(int y, int x, const string& str, int font_size = 15, const string& fill = "black",
              const string& dominant_baseline = "middle", const string& text_anchor = "middle") {
        stringstream ss;
        ss << "<text";
        ss << " x=\"" << x << "\" y=\"" << y << "\"";
        ss << " font-size=\"" << font_size << "\"";
        ss << " fill=\"" << fill << "\"";
        ss << " dominant-baseline=\"" << dominant_baseline << "\"";
        ss << " text-anchor=\"" << text_anchor << "\"";
        ss << ">" << str << "</text>";
        xmls.push_back(ss.str());
    }

    void image(int y, int x, int h, int w, const std::string& file_path) {
        std::stringstream ss;
        ss << "<image";
        ss << " x=\"" << x << "\" y=\"" << y << "\"";
        ss << " height=\"" << h << "\"";
        ss << " width=\"" << w << "\"";
        ss << " xlink:href=\"" << file_path << "\"";
        ss << "></image>";
        xmls.push_back(ss.str());
    }

    void clear() {
        xmls.clear();
    }

    string dump(const string& id = "") {
        stringstream ss;
        ss << "<svg ";
        if (id != "") ss << "id=\"" << id << "\" ";
        ss << "xmlns=\"http://www.w3.org/2000/svg\"";
        ss << " viewBox=\"0 0 " << W << " " << H << "\"";
        ss << " width=\"" << W << "\" height=\"" << H << "\">" << endl;

        for (const string& xml : xmls) {
            ss << xml << endl;
        }

        ss << "</svg>" << endl;
        return ss.str();
    }

    void save(const string& file_path, const string& id = "") {
        ofstream ofs(file_path);
        ofs << dump(id);
    }
};

javascriptで簡易アニメーション

  • ./svg/out0000.svgのように複数ファイルを./svg/以下に置いておく
  • 以下のhtmlを修正・用意して、ブラウザで開く
<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8" />
    <title>svg player</title>
    <style>
      #ui {
        margin: 5px;
      }
      #seekRange {
        width: 200px;
      }
      #seekNumber {
        width: 50px;
        height: 25px;
      }
      #speedNumber {
        width: 50px;
        height: 25px;
      }
      button {
        width: 50px;
        height: 30px;
      }
    </style>
  </head>
  <body>
    <div id="ui">
      <input id="seekRange" type="range" min="0" step="1" />
      <input id="seekNumber" type="number" min="0" step="1" />
      <button id="firstButton">|◀◀</button>
      <button id="prevButton">|◀</button>
      <button id="playButton"></button>
      <button id="nextButton">▶|</button>
      <button id="lastButton">▶▶|</button>
      <input
        id="speedNumber"
        type="number"
        min="1"
        max="30"
        step="1"
        value="8"
      />
    </div>
    <div id="field"></div>
    <script>
      // 使い方:
      // 1. ./svg/out0000.svgのように0から連番のsvgファイルをsvg/以下に置いておく
      // 2. fileNumを実際のファイル数に修正する
      // 3. htmlを開く
      const fileNum = 1000;

      let step = 0;
      let timer = null;

      // ファイルパス指定
      let svgs = [];
      for (let i = 0; i < fileNum; i++) {
        let idx = i.toString().padStart(4, "0");
        svgs.push("./svg/out" + idx + ".svg");
      }

      // サイズ指定
      let svg = document.createElement("img");
      document.getElementById("field").style.width = svg.width;
      document.getElementById("field").style.height = svg.height;
      document.getElementById("field").appendChild(svg);

      let seekRange = document.getElementById("seekRange");
      let seekNumber = document.getElementById("seekNumber");
      seekRange.max = fileNum - 1;
      seekNumber.max = fileNum - 1;

      //
      function setElement() {
        svg.src = svgs[step];
        seekRange.value = step;
        seekNumber.value = step;
      }

      function changeRange() {
        step = Number(seekRange.value);
        setElement();
      }

      function changeNumber() {
        step = Number(seekNumber.value);
        setElement();
      }

      function setFirst() {
        step = 0;
        setElement();
      }

      function setLast() {
        step = svgs.length - 1;
        setElement();
      }

      function inc() {
        if (step == svgs.length - 1) {
          if (timer != null) {
            playAndStop();
          }
          return;
        }
        step = (step + 1) % svgs.length;
        setElement();
      }

      function dec() {
        if (step == 0) return;
        step = (step - 1 + svgs.length) % svgs.length;
        setElement();
      }

      function playAndStop() {
        if (timer != null) {
          clearInterval(timer);
          timer = null;
          document.getElementById("playButton").textContent = "▶";
        } else {
          const speed = document.getElementById("speedNumber").value;
          if (step == svgs.length - 1) step = 0;
          timer = setInterval(inc, 1000 / speed);
          document.getElementById("playButton").textContent = "■";
        }
      }

      function changeSpeed() {
        if (timer == null) return;
        playAndStop(); // stop
        playAndStop(); // play
      }

      document
        .getElementById("seekRange")
        .addEventListener("change", changeRange);
      document
        .getElementById("seekNumber")
        .addEventListener("change", changeNumber);
      document.getElementById("nextButton").setAttribute("onclick", "inc()");
      document.getElementById("prevButton").setAttribute("onclick", "dec()");
      document
        .getElementById("firstButton")
        .setAttribute("onclick", "setFirst()");
      document
        .getElementById("lastButton")
        .setAttribute("onclick", "setLast()");

      document
        .getElementById("playButton")
        .setAttribute("onclick", "playAndStop()");
      document
        .getElementById("speedNumber")
        .addEventListener("change", changeSpeed);

      //
      setElement();
    </script>
  </body>
</html>

jupyter notebookでアニメーション

  • pip install ipywidgetsしておく
    • notebookは起動してた場合は再起動する
  • 1ファイルに各ターンのsvg<svg>〜</svg>を連続して書き込んだファイルを用意する
  • 以下のような感じでnotebook上で読み込んで表示する

# outs.svgの中身のイメージ
<svg>
...
</svg>

<svg>
...
</svg>

<svg>
...
</svg>
# 注意: display()を実行するたびに画像が一旦非表示になるような挙動をしてそうで、
# widgetを表示している以降のセルが少ないと、勝手にスクロールされてしまうっぽい。
# そのような場合は、widgetを実行するセルの下に多めに空セルを用意しておくとよいかも。
from IPython.display import display, SVG, HTML
from ipywidgets import Play, Layout, IntSlider, jslink, HBox, Button, interactive_output

# 複数のsvgを連続して出力してあるファイルを読み込む
with open("outs.svg") as f:
    svgs = list(map(lambda x: x + "</svg>", f.read().replace("\n","").split("</svg>")))
    del(svgs[-1])

# widget表示
def display_svg(turn):
    display(SVG(svgs[turn-1]))
play = Play(value=1, min=1, max=len(svgs), step=1, interval=200)
slider = IntSlider(1, 1, len(svgs), 1, layout=Layout(width="500px"))
jslink((play, "value"), (slider, "value"))

def increment_step(b):
    if slider.value + 1 < len(svgs):
        slider.value += 1
def decrement_step(b):
    if slider.value - 1 >= 0:
        slider.value -= 1
inc_button = Button(description="+",layout=Layout(width="50px"))
inc_button.on_click(increment_step)
dec_button = Button(description="-",layout=Layout(width="50px"))
dec_button.on_click(decrement_step)

ui = HBox([play, inc_button, dec_button, slider])
display(ui, interactive_output(display_svg, {"turn": slider}))

emscriptenで公式ビジュアライザのように動かす

  • emscriptenを使って、公式のビジュアライザのようにWebAssemblyにして動かせる
  • C++で書くことで、解答のC++コードと同じコードが使い回せる

環境構築

コード例

(間違った書き方してるかもしれないので注意)

// vis.cc
#include <emscripten/bind.h>
#include <emscripten/emscripten.h>

// ...

// input,outputに対してターンtの状態をsvg文字列で返す
string vis(const string& input, const string& output, int t) {
    string svg_text = "<svg> ... </svg>";
    return svg_text;
}

// 最大ターン数
int get_max_turn(const string& input, const string& output) {
    int turn = 100;
    return turn;
}

EMSCRIPTEN_BINDINGS(mod) {
    emscripten::function("vis", &vis);
    emscripten::function("get_max_turn", &get_max_turn);
}
<!-- index.html(細かいところは公式ビジュアライザを参考) -->
<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8" />
  </head>
  <body>
    <!-- フォームやdivなどを用意 -->
    <!-- ... -->
    <script type="module">
      // 描画を更新
      function visualize() {
        const input = document.getElementById("input").value;
        const output = document.getElementById("output").value;
        const t = Number(document.getElementById("turn").value);
        try {
          const ret = window.Module.vis(input, output, t);
          document.getElementById("result").innerHTML = ret;
        } catch (error) {
          console.log(error);
          document.getElementById("result").innerHTML = "<p>Invalid</p>";
        }
      }
      window.visualize = visualize; // window.hogeにいれることでグローバルでアクセスできる(html側から呼び出せる)

      // tの更新があった場合の処理
      function update_t(t) {
        const max_turn = Number(document.getElementById("turn").max);
        const new_turn = Math.min(Math.max(0, t), max_turn);
        document.getElementById("turn").value = new_turn;
        document.getElementById("t_bar").value = new_turn;
        visualize();
      }
      window.update_t = update_t;

      // アニメーションなどのコードを追加
      // ...
    </script>
    <script async type="text/javascript" src="vis.js"></script>
  </body>
</html>

ビルド

# vis.jsとvis.wasmができる
emcc -lembind vis.cc -O3 -o vis.js

# (iftreamなどで外部ファイルを読み込む場合(手元にin/*.txtを用意した場合))
emcc -lembind vis.cc -O3 -o vis.js --preload-file in/

利用方法(ローカル)

# vis.js, vis.wasm, index.htmlがあるディレクトリで以下を実行
python -m http.server 8000

# http://localhost:8000/ にアクセス

Tips

  • tooltipsを表示
    • <g><title>meg</title><rect>...</rect></g>のようにすると、マウスカーソルを載せたときにmsg情報を表示できる