Author: Kaz Aiso
Table of Contents
統合開発環境 Delphi / C++Builder で 3D表示のアプリを作る (無償で使えるStarter Editionでも可能)[JAPAN]
クロスプラットフォーム開発環境 Delphi / C++Builder を使用して かんたんに 3D表示のアプリを作成する手順をご紹介します。
この一つの手順で、iOS, Android, Windows, macOSで動作するアプリを作れる事が魅力。
アプリ内で3D表示を行うことができれば、3D のモデリングや、3D空間のシミュレーションを行うなど、新しいアプリの利用方法が広がります。例えば小さな一部屋内での家具の配置を試すようなアプリを作ったり、大きな町一つを丸ごと、バーチャルリアリティの世界として作る、といったことにも利用できるでしょう。
Delphi / C++Builder を使った 最も基本的な3Dアプリ作成のチュートリアルは、エンバカデロのDocwikiに「チュートリアル FireMonkey 3Dアプリケーションを作成する」という題名で記載されています。より詳しい情報を知りたい方は、こちらのDocwikiもご参考になってください。
3Dモデルを作ったアプリの作成は、無料で使えるStarter Editionでも、Windows 32bit向け限定ではありますが、作成できます。よっ太っ腹!
Delphi / C++Builder の Starter Edition はこちらの記事をご参考に
すでにDelphi / C++Builder / RAD Studioをお持ちの方は、そのお持ちの開発環境でお試しください。お持ちでない方はトライアル版使って当記事内容を確認できます。トライアル版導入については、こちらを記事をご参照あれ
作成手順のビデオ
作成手順のビデオが公開されています。こちらもどうぞ
3D地球儀アプリの作成手順
今回はいつもの、2Dアプリケーションフォームの上に、3D表示できる 3DViewPort といういわゆるレイヤーを張って、その中でライティングをしたり、3Dのブツを置いたり、3Dのブツに絵を張り付けて地球儀に仕立てあげ、かつその地球儀をアニメーションでくるくる回したりしてみます。
では実際の作成手順。
- Delphi / C++Builderを起動し、最初に表示される ウエルカムページの 「マルチデバイスアプリケーション Delphi (C++Builder)の新規作成」を選択
-
「マルチデバイスアプリケーション」のダイアログが現れたら「空のアプリケーション」を選択して[OK]ボタンをクリック
※ここでは「空のアプリケーション」(通常の2Dフォーム)を選択していますが、その右横の「3Dアプリケーション」を選んで、最初から3D表示のアプリを作ることができます。この場合、後述する 「3DViewPort」コンポーネントを張り付けずとも3DのShapesなどを表示することができます。最初から3Dアプリケーションフォームを使用している場合は、逆に、3Dフォームに対して、2Dコンポーネントを張り付けるための「TLayer3D」コンポーネントがあります。このコンポーネントの中に2Dコンポーネントたちを表示することができます。
今回の手順においては、一般的なアプリの作り方として、2Dフォームを使って、その中に3D表示レイヤー(T3DVirePort)をもち、3D世界を表現する方法をとっています 。 -
3Dアイテムを表示するコンポーネントとして「TViewPort3D」を使用します。右下のツールパレットからフォーム上にドラッグアンドロップします。
-
TViewPort3Dは「ViewPort3D1」という名称のコンポーネントとしてフォーム上に配置されます。この「ViewPort3D1」のプロパティのAlignにClientを設定して、ViewPort3D1の表示領域をフォーム全体へと引き延ばしておきましょう。
-
ついでに宇宙空間ぽく、色も変えます。プロパティのColorにBlackを設定しておきましょう。
※この「ViewPort3D1」内に3Dアイテムたちを配置できます。
-
3Dのアイテムたちはツールパレットの3D関連のグループに収められています。まずは 3D Sceneグループ内の「TLight」をViewPort3D1上にドラッグアンドドロップします
※これは3Dのエリア内で、どの方向から光が照らされているか、を設定できるいわゆる光源を表しています。地球儀においては太陽の光の代わりです。今回はプロパティのRotation Angle のY軸に45を設定し、ななめ45度の水平方向から光が照らされているように設定してみましょう。
-
「Light1」をクリックして選択し、左下のプロパティのRotationAngleの左横に表示されている[+]マークをクリックすると、X,Y,Zのサブアイテムが展開されます。このYに [45]を設定します。
ちなみにプロパティの「LightType」で一定方向からの照射を表すDirectionalや、点光源、スポットタイプなどを設定できます。今回はそのままDirectionalを使用します。
-
Sphere1のプロパティのDepthに[10]、Heightに[10]、Widthに[10]を設定して球体をより大きく変更しておきます。
3Dのモデルにくっついている水色のツマミをマウスでドラッグすることでも大きさを変更できます。今回は、きれいな球体をマウスで整えるのは困難なため、プロパティのDepth、Height、Widthに直接数値を入力して大きさを変更しました。
-
地球儀は地軸が23.43傾いていますので、この球体 Shpere1 のプロパティ「RotationAngle.Z」軸も[23.43]ずらしておきましょう。
-
今度は、この赤い球体に地球儀のテクスチャ、つまり絵ですね。これを張って地球儀らしくしていきます。Materialsグループ内の「TLightMaterialSource」をドロップします。
-
これを使うと、光源の影響をうけた明暗のある素材を張り付けることができます。LightMaterialSource1の プロパティ「Texture」の[BitmapEmpty]と表示されている部分を選択、ダブルクリックすると、「ビットマップエディタ」ダイアログが開きます。ここで「読み込み」ボタンをクリックして貼り付けたい写真、絵を選択してOKをクリックします。
今回、NASAで無償提供している地球の2D画像を利用しました。画像データはこちらで配布されております。
ちなみに、雲の画像データも配布されており、これを透過GIFなどにして、この地球儀Sphere1の上にさらにSphereを子として載せて、雲のレイヤーとして表示する、などの発展もできます。雲の画像データはこちら -
これで、LightMaterialSource1に地球儀の絵となる部分が読み込まれました。これを球体に張り付ける素材として登録します。Sphere1をクリックして選択して、プロパティの[MaterialSource]に先ほどの「LightMaterialSource1」を設定します。これでななめ45から光が照射されている陰影のある地球儀ができあがりました。
-
球体が若干いびつです。もうすこし、細やかに表現するためにプロパティのSubdivisionsAxesを[32]、 SubdivisionsHeightを[32]に変更します。この数値を変更することでより細やかに3Dを表現できます。
-
次に、この地球儀をアニメーションさせて回します。 TFloatAnimationを使ってこの球体のY軸を継続的に変えてあげればよいのです。ツールパレットからTFloatAnimationをSphere1の上にドロップします。「構造ビュー」での表示ではこんな感じに、Sphere1の子としてFloatAnimation1が登録されます。
-
FloatAnimation1のプロパティの「Duration]に[60]秒、「Enable」を[True]を設定してアニメーションを有効に、「Loop」を[True]にして繰り返しとして、「PropertyName」に[RotationAngle.Y]を指定して、水平方向にY軸の値をアニメーションするように設定します。
-
さらに「StartFromCurrent」 を[True]にして現在の値からアニメーションをスタートするように設定し、「StopValue」に[-360]として、球体の1回転分の値を設定します。
-
またイベントハンドラの記述ももちろん可能です。Sphere1がクリックされたときに、ライティングの角度を変えるコードを記述してみましょう。左上の「構造ビュー」に表示されている 「Sphere1」をダブルクリックすると、Sphere1のクリック時のイベントハンドラのコード記述部にジャンプします。ここに TAnimaterクラスのAnimationFloatを使ってLight1のRotationAngle.Yの値を45度ずつ、0.5秒間でアニメーションするように記述します。
1 2 3 4 5 |
<span class="c1" style="color:#999988;font-style:italic;">//Delphiのコード</span> <span class="k" style="font-weight:bold;">procedure</span> <span class="nc" style="color:#445588;font-weight:bold;">TForm1</span><span class="o" style="font-weight:bold;">.</span><span class="nf" style="color:#990000;font-weight:bold;">Sphere1Click</span><span class="p">(</span><span class="n">Sender</span><span class="o" style="font-weight:bold;">:</span> <span class="kt" style="color:#445588;font-weight:bold;">TObject</span><span class="p">)</span><span class="o" style="font-weight:bold;">;</span> <span class="k" style="font-weight:bold;">begin</span> <span class="n">TAnimator</span><span class="o" style="font-weight:bold;">.</span><span class="n">AnimateFloat</span><span class="p">(</span><span class="n">Light1</span><span class="o" style="font-weight:bold;">,</span> <span class="s" style="color:#d01040;">'RotationAngle.Y'</span><span class="o" style="font-weight:bold;">,</span> <span class="n">Light1</span><span class="o" style="font-weight:bold;">.</span><span class="n">RotationAngle</span><span class="o" style="font-weight:bold;">.</span><span class="n">Y</span><span class="o" style="font-weight:bold;">-</span><span class="mi" style="color:#009999;">45</span><span class="o" style="font-weight:bold;">,</span> <span class="mf" style="color:#009999;">0.5</span><span class="p">)</span><span class="o" style="font-weight:bold;">;</span> <span class="k" style="font-weight:bold;">end</span><span class="o" style="font-weight:bold;">;</span> |
1 2 3 4 5 6 7 |
<span class="c1" style="color:#999988;font-style:italic;">//C++Builderのコード</span> <span class="c1" style="color:#999988;font-style:italic;">//---------------------------------------------------------------------------</span> <span class="kt" style="color:#445588;font-weight:bold;">void</span> <span class="kr" style="font-weight:bold;">__fastcall</span> <span class="n">TForm1</span><span class="o" style="font-weight:bold;">::</span><span class="n">Sphere1Click</span><span class="p">(</span><span class="n">TObject</span> <span class="o" style="font-weight:bold;">*</span><span class="n">Sender</span><span class="p">)</span> <span class="p">{</span> <span class="n">TAnimator</span><span class="o" style="font-weight:bold;">::</span><span class="n">AnimateFloat</span><span class="p">(</span><span class="n">Light1</span><span class="p">,</span> <span class="s" style="color:#d01040;">"RotationAngle.Y"</span><span class="p">,</span> <span class="n">Light1</span><span class="o" style="font-weight:bold;">-></span><span class="n">RotationAngle</span><span class="o" style="font-weight:bold;">-></span><span class="n">Y</span><span class="o" style="font-weight:bold;">-</span><span class="mi" style="color:#009999;">45</span><span class="p">,</span> <span class="mf" style="color:#009999;">0.5</span><span class="p">);</span> <span class="p">}</span> <span class="c1" style="color:#999988;font-style:italic;">//---------------------------------------------------------------------------</span> |
これで実行[F9]キーを行うと、60秒描けて1周する3D地球儀が表示されます。クリックしてみると、ライティングが45度づつ変わっていきます。(動作イメージはこのブログ冒頭に引用している動画をご参照ください)
それでは3Dアプリの作成をお楽しみください。
おわりに
ということで、3D表示アプリを作ってみました。
コンポーネントが用意されていて、結構かんたんにつくれてしまいます。
ここで紹介している手順より手を加えておりますが、Delphi版のソースコード(プロジェクト)をgithubに上げてあります。ご参考まで。
https://github.com/kazaiso/3DSphere
動作イメージ:
関連資料・参考資料
先人の方々には頭が下がります。ありがとうございます。
- 「誰でも始められるカンタン FireMonkey 3D 入門 : 慶應義塾大学 大学院理工学研究科 藤代研究室 研究生 中山雅紀 様
- FireMonkey で作る地球儀 ~ 多分、4 分以内に ~ : Hideaki Tominaga 様 (Youtube)
Design. Code. Compile. Deploy.
Start Free Trial Upgrade Today
Free Delphi Community Edition Free C++Builder Community Edition