ProcessingAdventCalendar
今年もProcessingAdventCalendar2015※1ATNDで開催してます(https://atnd.org/events/72012)が開催されております。一応主催者側なのですが、たまには記事を1つぐらい書こうかと思い、参加する運びとなりました。
12/27の予定でしたが、諸事情で本日の担当となりました。
今回の成果物
3Dプリンタを手に入れて数カ月がたちました。プログラミングで3Dモデルを作って、それを3Dプリンタで出力する、という目標を持っていたのですが、このたびようやく形になりました。やったねヽ(・-・)ノ

概要
Processing※2メディアアート向けプログラミング言語環境(https://processing.org/)で3Dモデルを生成し、これを3Dプリンタで出力します。お手軽です。
過去にはカッティングプロッタのCraftRobo※3プリンタにカッターが付いたようなもの。紙を好きな形に切ることができます(http://craftrobo.jp/)と連携したこともあります。(ProcessingとCraftROBOで、クリスマスカードを作成する)
ディスプレイに表示したり、プリンタで紙に印刷する以外にもいろいろなアウトプットの楽しみがあります。個人でも気軽に手出しができるものが増えてきてうれしいものですね。
今回使用したプリンタはXYZwareのダヴィンチ Jr 1.0※4ダヴィンチ Jr 1.0(https://jp.xyzprinting.com/product/da-Vinci-1.0-Junior)になります。素材はPLA樹脂というプラスチックの素材です。積層式ですし、どうしても細かいところは荒くなってしまいますが、こうやって手軽に遊ぶのには最適です。
時間は1時間程度、コストは単純計算で約30円でした。
技術編
ソースコードは後半に貼ってあります。
ライフゲーム生成
3Dモデルを生成するわけですが、せっかくプログラミングで作るわけですから、ありきたりな形ではつまらないものです。そこでライフゲーム※5ライフゲーム(https://ja.wikipedia.org/wiki/%E3%83%A9%E3%82%A4%E3%83%95%E3%82%B2%E3%83%BC%E3%83%A0)を3D空間に描画し、それを基にモデルを生成するという方針にしました。
ライフゲームの作り方はここでは割愛します。
ちなみに各ボックスの間を開けているのは、3Dプリンタと相性が悪かったためです。うまく凹みが認識できないらしく。。
ほんとはくっつけたかったのですが、ここは妥協です。
マウスでモデルを移動したり回転したりして、隅々まで見れるようにしています。
ちなみにですが、3Dプリンタで出力するためには、(当たり前ですが)ポリゴンの内側は密閉されている必要があります。
オブジェクトファイル出力
描画した3Dオブジェクトを3Dファイルに変換するライブラリとして、NervousSystemのOBExport※6NervousSystem(http://n-e-r-v-o-u-s.com/tools/obj/)を使用いたしました。Processing 3.0系であればライブラリのインポートメニューから、「obj」を検索すれば見つかります。

使い方は簡単で、インポートして、開始のbeginRecord()と終了のendRecord()を適当な位置に組み込めばOKです。
beginRecord()~endRecord()間で描画したポリゴンが指定したファイルに記録されます。
1 |
import nervoussystem.obj.*; |
1 |
beginRecord("nervoussystem.obj.OBJExport", "v2.obj"); |
1 |
endRecord(); |
STL変換
今回使用する3Dプリンタ用のソフトXYZwareは、objファイルを読むことができません。しかしstlファイルを読むことができるため、obj→stl変換を行います。
変換ソフトとしてBlender※7Blender(http://www.blender.org/)を使用します。




3Dプリンタ出力
XYZwareで、作成したstlファイルを読み込み、3Dプリンタへ出力します。

印刷風景
iPhoneのタイムラプス機能で撮影しました。なんでもiPhoneでできる便利な世の中です。
完成

※Processing上は逐次ライフゲームが進むため、出力ファイルと実行画面で模様が違うのはご了承ください。
ソースコード
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 |
import nervoussystem.obj.*; // obj変換ライブラリ final int X_MAX = 20; // 横セル数 final int Y_MAX = 20; // 縦セル数 float[][] cells = new float[X_MAX][Y_MAX]; // セル boolean record = false; // obj変換フラグ final int EDGE = 10; // セル辺の長さ final int HEIGHT = -3; // 底面の高さ //-------------------------------------------------- // setup //-------------------------------------------------- void setup() { size(800, 800, P3D); // 各セルを初期化 for (int y = 0; y < Y_MAX; y++) { for (int x = 0; x < X_MAX; x++) { cells[x][y] = 0; } } } //-------------------------------------------------- // draw //-------------------------------------------------- void draw() { // 描画モード切り替え(obj変換時は座標変換をするとobjの座標もずれるので。) if (record) { beginRecord("nervoussystem.obj.OBJExport", "v2.obj"); } else { background(0); directionalLight(255, 255, 255, 1, 1, -1); ambientLight(32, 32, 32); translate(width / 2 - EDGE * X_MAX / 2, height / 2 - EDGE * Y_MAX / 2); translate(0, 0, mouseX); rotateX(radians(-mouseY)); } fill(255); noStroke(); // ライフゲーム処理 life(); // 底面の描画 base(); // ライフゲームの結果をボックスで出力 for (int y = 1; y < Y_MAX - 1; y++) { for (int x = 1; x < X_MAX - 1; x++) { if(cells[x][y] == 1){ box(x, y, 5); } } } // obj変換中なら終了処理 if (record) { endRecord(); print("recorded"); record = false; } } //-------------------------------------------------- // ボックス描画 //-------------------------------------------------- void box(int x, int y, float z) { // 壁面 beginShape(TRIANGLE_STRIP); vertex(x * EDGE + 0 , y * EDGE + 0, 0); vertex(x * EDGE + 0 , y * EDGE + 0, z); vertex(x * EDGE + EDGE-1, y * EDGE + 0, 0); vertex(x * EDGE + EDGE-1, y * EDGE + 0, z); vertex(x * EDGE + EDGE-1, y * EDGE + EDGE-1, 0); vertex(x * EDGE + EDGE-1, y * EDGE + EDGE-1, z); vertex(x * EDGE + 0 , y * EDGE + EDGE-1, 0); vertex(x * EDGE + 0 , y * EDGE + EDGE-1, z); vertex(x * EDGE + 0 , y * EDGE + 0, 0); vertex(x * EDGE + 0 , y * EDGE + 0, z); endShape(); // 上面 beginShape(TRIANGLE_STRIP); vertex(x * EDGE + 0, y * EDGE + 0, z); vertex(x * EDGE + EDGE-1, y * EDGE + 0, z); vertex(x * EDGE + 0, y * EDGE + EDGE-1, z); vertex(x * EDGE + EDGE-1, y * EDGE + EDGE-1, z); endShape(); } //-------------------------------------------------- // ライフゲーム処理 //-------------------------------------------------- void life() { float t[][] = new float[X_MAX][Y_MAX]; // テンポラリ // テンポラリを初期化 for (int y = 0; y < Y_MAX; y++) { for (int x = 0; x < X_MAX; x++) { t[x][y] = 0; } } // テンポラリに次の世代を書き込む for (int y = 1; y < Y_MAX - 1; y++) { for (int x = 1; x < X_MAX - 1; x++) { int result = count(x, y); if (result == 1 || result == 3) { t[x][y] = 0; } else { t[x][y] = 1; } } } // テンポラリを適用 cells = t; } //-------------------------------------------------- // 自セルの周りの生存セルの数を計算する //-------------------------------------------------- int count(int x, int y) { int result = 0; result += cells[x-1][y-1] == 1 ? 0 : 1; result += cells[x+0][y-1] == 1 ? 0 : 1; result += cells[x+1][y-1] == 1 ? 0 : 1; result += cells[x-1][y+0] == 1 ? 0 : 1; result += cells[x+1][y+0] == 1 ? 0 : 1; result += cells[x-1][y+1] == 1 ? 0 : 1; result += cells[x+0][y+1] == 1 ? 0 : 1; result += cells[x+1][y+1] == 1 ? 0 : 1; return result; } //-------------------------------------------------- // 底面の描画 //-------------------------------------------------- void base(){ // 上面 beginShape(TRIANGLE_STRIP); vertex(0, 0, 0); vertex(X_MAX * EDGE, 0, 0); vertex(0, Y_MAX * EDGE, 0); vertex(X_MAX * EDGE, Y_MAX * EDGE, 0); endShape(); // 下面 beginShape(TRIANGLE_STRIP); vertex(0, 0, HEIGHT); vertex(X_MAX * EDGE, 0, HEIGHT); vertex(0, Y_MAX * EDGE, HEIGHT); vertex(X_MAX * EDGE, Y_MAX * EDGE, HEIGHT); endShape(); // 壁面 beginShape(TRIANGLE_STRIP); vertex(0, 0, HEIGHT); vertex(0, 0, 0); vertex(0, Y_MAX * EDGE, HEIGHT); vertex(0, Y_MAX * EDGE, 0); vertex(X_MAX * EDGE, Y_MAX * EDGE, HEIGHT); vertex(X_MAX * EDGE, Y_MAX * EDGE, 0); vertex(X_MAX * EDGE, 0, HEIGHT); vertex(X_MAX * EDGE, 0, 0); vertex(0, 0, HEIGHT); vertex(0, 0, 0); endShape(); } // マウスクリックしたら、obj出力モードにする。 void mousePressed() { record = true; } |
感想
もっと高価な3Dプリンタなら綺麗に出力できたり、凹みも問題なくできたりするのでしょうね。
今後もコンピュータならではのモデルを生成して創っていきたいものです。
皆様も素敵なProcessingライフをどうぞ。
脚注
↑戻る1 | ATNDで開催してます(https://atnd.org/events/72012) |
↑戻る2 | メディアアート向けプログラミング言語環境(https://processing.org/) |
↑戻る3 | プリンタにカッターが付いたようなもの。紙を好きな形に切ることができます(http://craftrobo.jp/) |
↑戻る4 | ダヴィンチ Jr 1.0(https://jp.xyzprinting.com/product/da-Vinci-1.0-Junior) |
↑戻る5 | ライフゲーム(https://ja.wikipedia.org/wiki/%E3%83%A9%E3%82%A4%E3%83%95%E3%82%B2%E3%83%BC%E3%83%A0) |
↑戻る6 | NervousSystem(http://n-e-r-v-o-u-s.com/tools/obj/) |
↑戻る7 | Blender(http://www.blender.org/) |
“ライフゲームを3Dプリンタで出力する” への1件のフィードバック