OSDN Git Service

進行描画スレッドを廃止し、GUIスレッドで RenderLoop を組むように変更。
authorくまかみ工房 <kumakamikoubou@gmail.com>
Thu, 3 Nov 2016 14:46:12 +0000 (23:46 +0900)
committerくまかみ工房 <kumakamikoubou@gmail.com>
Thu, 3 Nov 2016 14:46:12 +0000 (23:46 +0900)
処理速度はあまり変わらず、それでいてスレッド間同期の手間が省ける。

FDK24/ApplicationBase.cs [deleted file]
FDK24/FDK24.csproj
FDK24/Log.cs
FDK24/メディア/デバイスリソース.cs
FDK24/メディア/文字列画像.cs
StrokeStyleT/Program.cs
StrokeStyleT/StrokeStyleT.cs
StrokeStyleT/StrokeStyleT.csproj
StrokeStyleT/Win32.cs [new file with mode: 0644]

diff --git a/FDK24/ApplicationBase.cs b/FDK24/ApplicationBase.cs
deleted file mode 100644 (file)
index bf69618..0000000
+++ /dev/null
@@ -1,334 +0,0 @@
-using System;
-using System.Collections.Generic;
-using System.ComponentModel;
-using System.Diagnostics;
-using System.Linq;
-using System.Threading;
-using System.Threading.Tasks;
-using System.Windows.Forms;
-using Microsoft.VisualBasic.ApplicationServices;
-
-namespace FDK
-{
-       /// <summary>
-       /// アプリケーションフォームの基礎クラス。
-       /// </summary>
-       public class ApplicationBase : Microsoft.VisualBasic.ApplicationServices.WindowsFormsApplicationBase
-       {
-               /// <summary>
-               /// 基本的に、このインスタンスを使用する任意のスレッドはこれを lock してからアクセスすること。
-               /// </summary>
-               protected readonly object スレッド間同期 = new object();
-
-               public bool 全画面モードである
-               {
-                       get;
-                       protected set;
-               } = false;
-               public bool ウィンドウモードである
-               {
-                       get { return !this.全画面モードである; }
-                       protected set { this.全画面モードである = !value; }
-               }
-
-               public ApplicationBase() : base()
-               {
-                       this.EnableVisualStyles = true;
-                       this.IsSingleInstance = true;   // 二重起動を禁止する
-                       this.MainForm = new Form();
-                       this.MainForm.Load += OnLoad;
-                       this.MainForm.FormClosing += OnClosing;
-                       this.MainForm.ClientSizeChanged += OnClientSizeChanged;
-               }
-
-               protected SharpDX.Size2F 設計画面サイズdpx = SharpDX.Size2F.Empty; // 初期化する() 内で設定すること。
-               protected FDK.メディア.デバイスリソース デバイスリソース = null;
-               protected System.Threading.Thread 進行描画スレッド = null;
-               protected AutoResetEvent 進行描画スレッドのキャンセル = null;
-               protected System.Threading.AutoResetEvent 進行描画スレッドの起動を完了した = new System.Threading.AutoResetEvent( false );
-
-               protected virtual void 初期化する()
-               {
-                       //----------------
-                       // 以下は実装例。
-                       //----------------
-                       lock( this.スレッド間同期 )
-                       {
-                               Debug.Assert( null == this.デバイスリソース, "デバイスリソースの作成前であること。" );
-                               this.設計画面サイズdpx = new SharpDX.Size2F( 640, 480 );
-                       }
-               }
-               protected virtual void 終了する()
-               {
-                       //----------------
-                       // 以下は実装例。
-                       //----------------
-                       lock( this.スレッド間同期 )
-                       {
-                               Debug.Assert( null != this.デバイスリソース, "デバイスリソースが解放される前であること。" );
-                       }
-               }
-               protected virtual void シーンを描画する()
-               {
-                       //----------------
-                       // 以下は実装例。
-                       // このメソッドはGUIスレッドではなく進行描画スレッドから呼び出されるので注意する。
-                       //----------------
-
-                       lock( this.スレッド間同期 )
-                       {
-
-                               // ここで、描画を行う。
-                               // ...
-
-
-                               // ここで、入力を行う。
-                               // ...
-                       }
-
-                       // 表示する。垂直帰線待ちなどで時間がかかるので、lock しないこと。
-                       this.デバイスリソース.SwapChain1.Present( 0, SharpDX.DXGI.PresentFlags.None );
-               }
-               protected virtual void デバイス依存リソースを解放する()
-               {
-                       lock( this.スレッド間同期 )
-                       {
-                               Debug.Assert( null != this.デバイスリソース );  // 解放前であること。
-
-                               // ここで自分のデバイス依存リソースを解放する。
-                               // ...
-                       }
-               }
-               protected virtual void デバイス依存リソースを再構築する()
-               {
-                       lock( this.スレッド間同期 )
-                       {
-                               Debug.Assert( null != this.デバイスリソース );  // 再生成済みであること。
-
-                               // ここで自分のデバイス依存リソースを解放する。
-                               // ...
-
-                       }
-               }
-               protected void 全画面モードとウィンドウモードを切り替える()
-               {
-                       lock( this.スレッド間同期 )
-                       {
-                               if( this.全画面モードである )
-                               {
-                                       this.デバイスリソース.SwapChain1.SetFullscreenState( false, null );  // ウィンドウモードへ。
-                                       this.ウィンドウモードである = true;
-                               }
-                               else
-                               {
-                                       this.デバイスリソース.SwapChain1.SetFullscreenState( true, null );    // 全画面モードへ。
-                                       this.全画面モードである = true;
-                               }
-                       }
-               }
-               /// <summary>
-               /// アプリが二重起動されたときに発生するイベント。
-               /// </summary>
-               /// <remarks>
-               /// 後続のインスタンスは起動せず、既存のインスタンスに対してこのイベントが発生する。
-               /// eventArg.CommandLine で、後続のインスタンスのコマンドライン引数を確認することができる。
-               /// </remarks>
-               protected override void OnStartupNextInstance( StartupNextInstanceEventArgs eventArgs )
-               {
-                       // 必要がれば、派生クラスで実装すること。
-               }
-
-               private bool Closing中 = false;
-
-               private void OnLoad( object sender, EventArgs e )
-               {
-                       FDK.Log.現在のスレッドに名前をつける( "GUI" );
-                       FDK.Log.BeginInfo( $"{FDK.Utilities.現在のメソッド名}" );
-
-                       #region " アプリケーションを初期化する。"
-                       //----------------
-                       FDK.Log.BeginInfo( "派生クラスを初期化します。" );
-
-                       this.初期化する();
-                       Debug.Assert( SharpDX.Size2F.Empty != this.設計画面サイズdpx, "初期化メソッド内で設計画面サイズを設定してあること。" );
-
-                       FDK.Log.Info( $"設計画面サイズ: {this.設計画面サイズdpx}" );
-                       FDK.Log.Info( $"物理画面サイズ: {this.MainForm.ClientSize}" );
-
-                       FDK.Log.EndInfo( "派生クラスを初期化しました。" );
-                       //----------------
-                       #endregion
-                       #region " デバイスリソースを作成する。"
-                       //----------------
-                       FDK.Log.BeginInfo( "デバイスリソースを作成します。" );
-                       lock( this.スレッド間同期 )
-                       {
-                               this.デバイスリソース = new メディア.デバイスリソース();
-                               this.デバイスリソース.設計画面サイズdpx = this.設計画面サイズdpx;
-                               this.デバイスリソース.すべてのリソースを作成する( this.MainForm.ClientSize, this.MainForm.Handle );
-                       }
-                       FDK.Log.EndInfo( "デバイスリソースを作成しました。" );
-                       //----------------
-                       #endregion
-                       #region " 進行描画スレッドを開始する。"
-                       //----------------
-                       FDK.Log.BeginInfo( "進行描画スレッドを開始します。" );
-                       lock( this.スレッド間同期 )
-                       {
-                               this.進行描画スレッドのキャンセル = new AutoResetEvent( false );
-                               this.進行描画スレッド = new Thread( this.進行描画スレッドエントリ );
-                               //this.進行描画スレッド.Priority = ThreadPriority.AboveNormal;
-                               this.進行描画スレッド.Start();
-                               this.進行描画スレッドの起動を完了した.WaitOne();        // スレッドの起動完了通知を待つ。
-                       }
-                       //----------------
-                       #endregion
-
-                       FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
-               }
-               private void OnClosing( object sender, FormClosingEventArgs e )
-               {
-                       FDK.Log.BeginInfo( $"{FDK.Utilities.現在のメソッド名}" );
-
-                       this.Closing中 = true;
-
-                       // 進行描画スレッドを終了する。
-                       if( ( null != this.進行描画スレッドのキャンセル ) &&
-                               ( null != this.進行描画スレッド ) )
-                       {
-                               try
-                               {
-                                       this.進行描画スレッドのキャンセル.Set();  // lock 内で呼び出したら絶対タイムアウトになるので注意。
-                                       FDK.Log.Info( "進行描画スレッドにキャンセルを発行しました。" );
-
-                                       bool done = this.進行描画スレッド.Join( millisecondsTimeout: 5000 );    // タイムアウトは保険。
-                                       if( done )
-                                               FDK.Log.Info( "進行描画スレッドの完了を確認しました。" );
-                                       else
-                                               FDK.Log.ERROR( "進行描画スレッドの完了を時間内に確認できませんでした。" );
-                               }
-                               catch( AggregateException ex )
-                               {
-                                       if( ex.InnerExceptions.Any( ( 内部例外 ) => ( 内部例外 is OperationCanceledException ) ) )
-                                       {
-                                               // OK
-                                       }
-                                       else
-                                       {
-                                               throw;  // NG
-                                       }
-                               }
-                               catch( ObjectDisposedException )
-                               {
-                                       // タスクがすでに終わってた。
-                               }
-                       }
-
-                       lock( this.スレッド間同期 )
-                       {
-                               // 派生クラスの終了処理を呼び出す。
-                               this.終了する();
-
-                               // デバイスリソースを解放する。
-                               this.デバイスリソース?.Dispose();
-                               this.デバイスリソース = null;
-                       }
-
-                       FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
-               }
-               private void OnClientSizeChanged( object sender, EventArgs e )
-               {
-                       FDK.Log.BeginInfo( $"{FDK.Utilities.現在のメソッド名}" );
-                       FDK.Log.Info( $"新しいクライアントサイズ = {this.MainForm.ClientSize}" );
-
-                       lock( this.スレッド間同期 )
-                       {
-                               #region " 実行条件チェック。"
-                               //----------------
-                               if( null == this.デバイスリソース )
-                               {
-                                       FDK.Log.Info( " まだ初期化されてないので、何もしません。" );
-                                       FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
-                                       return;
-                               }
-                               if( this.MainForm.WindowState == FormWindowState.Minimized )
-                               {
-                                       FDK.Log.Info( "最小化されました。" );
-                                       FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
-                                       return; // 何もしない
-                               }
-                               if( this.Closing中 )
-                               {
-                                       FDK.Log.Info( " 終了処理中なので、何もしません。" );
-                                       FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
-                                       return;
-                               }
-                               //----------------
-                               #endregion
-
-                               var dr = this.デバイスリソース;
-                               Debug.Assert( null != dr, "デバイスリソースが作成済みであること。" );
-
-                               // 現在の画面モードを取得しておく。(Alt+TABなど、勝手に全画面を解除されることもあるので。)
-                               SharpDX.Mathematics.Interop.RawBool fullscreen;
-                               SharpDX.DXGI.Output outputTarget;
-                               dr.SwapChain1.GetFullscreenState( out fullscreen, out outputTarget );
-                               this.全画面モードである = fullscreen;
-                               outputTarget?.Dispose();
-                               FDK.Log.Info( $"現在、全画面モードである = {this.全画面モードである}" );
-
-                               // (1) リソースを解放して、
-                               this.デバイス依存リソースを解放する();
-                               dr.サイズに依存するリソースを解放する();
-
-                               // (2) 物理画面サイズを変更して、
-                               dr.物理画面サイズpx = new SharpDX.Size2F( this.MainForm.ClientSize.Width, this.MainForm.ClientSize.Height );
-
-                               // (3) リソースを再構築する。
-                               dr.サイズに依存するリソースを作成する();
-                               this.デバイス依存リソースを再構築する();
-                       }
-
-                       FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
-               }
-
-               private void 進行描画スレッドエントリ()
-               {
-                       FDK.Log.現在のスレッドに名前をつける( "Main" );
-                       FDK.Log.Info( "進行描画スレッドを起動しました。" );
-
-                       this.進行描画スレッドの起動を完了した.Set();    // 生成元へ起動完了を通知する。
-
-                       while( true )
-                       {
-                               bool アプリを終了せよ = false;
-
-                               lock( this.スレッド間同期 )
-                               {
-                                       // 別スレッドからキャンセル要求があれば、終了フラグを立てる。
-                                       if( this.進行描画スレッドのキャンセル.WaitOne( 0 ) )
-                                       {
-                                               アプリを終了せよ = true;
-                                       }
-                                       else
-                                       {
-                                               // D3Dデバイスが消失していれば再構築する。
-                                               bool 異常発生 = false;
-                                               this.デバイスリソース.D3Dデバイスが消失していれば再構築する( out 異常発生 );
-                                               if( 異常発生 )
-                                                       アプリを終了せよ = true;
-                                       }
-                               }
-
-                               // 終了フラグがセットされていれば、ループを抜けてスレッドを終了する。
-                               if( アプリを終了せよ )
-                                       break;
-
-                               // それ以外なら、シーンを進行・描画する。
-                               this.シーンを描画する();
-                       }
-
-                       FDK.Log.Info( "進行描画スレッドを終了します。" );
-               }
-       };
-}
index e57a765..bf1e084 100644 (file)
@@ -36,7 +36,6 @@
       <HintPath>..\packages\CSCore.1.1.0\lib\net35-client\CSCore.dll</HintPath>
       <Private>True</Private>
     </Reference>
-    <Reference Include="Microsoft.VisualBasic" />
     <Reference Include="SharpDX, Version=3.1.1.0, Culture=neutral, PublicKeyToken=b4dcf0f35e5521f1, processorArchitecture=MSIL">
       <HintPath>..\packages\SharpDX.3.1.1\lib\net45\SharpDX.dll</HintPath>
       <Private>True</Private>
@@ -86,7 +85,6 @@
   </ItemGroup>
   <ItemGroup>
     <Compile Include="Activity.cs" />
-    <Compile Include="ApplicationBase.cs" />
     <Compile Include="FDKException.cs" />
     <Compile Include="Log.cs" />
     <Compile Include="Memory.cs" />
     </EmbeddedResource>
     <EmbeddedResource Include="メディア\テクスチャ用シェーダコード.hlsl" />
   </ItemGroup>
+  <ItemGroup />
   <Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
   <!-- To modify your build process, add your task inside one of the targets below and uncomment it. 
        Other similar extension points exist, see Microsoft.Common.targets.
index 6b3d8e0..6080221 100644 (file)
@@ -53,14 +53,14 @@ namespace FDK
                                Log.一定時間が経過していたら区切り線を表示する();
                                Trace.TraceInformation( $"{Log.日時とスレッドID}{Log.インデックスを返す( Log.深さ )}{開始ブロック名} --> 開始" );
 
-                               Log.深さ++;
+                               Log.深さ += 4;
                        }
                }
                public static void EndInfo( string 終了ブロック名 )
                {
                        lock( Log.排他 )
                        {
-                               Log.深さ = Math.Max( --Log.深さ, 0 );
+                               Log.深さ = Math.Max( ( Log.深さ - 4 ), 0 );
 
                                Log.一定時間が経過していたら区切り線を表示する();
                                Trace.TraceInformation( $"{Log.日時とスレッドID}{Log.インデックスを返す( Log.深さ )}{終了ブロック名} <-- 終了" );
index 447196d..76599ca 100644 (file)
@@ -8,7 +8,7 @@ namespace FDK.メディア
        public class デバイスリソース : IDisposable
        {
                public SharpDX.Size2F 設計画面サイズdpx = new SharpDX.Size2F( 1920f, 1080f );    // 設計サイズ。
-               public SharpDX.Size2F 物理画面サイズpx = new SharpDX.Size2F( 0, 0 );   // (0, 0) は、サイズ依存リソース無効の印。
+               public SharpDX.Size2F 物理画面サイズpx = new SharpDX.Size2F( 0f, 0f );   // (0, 0) は、サイズ依存リソース無効の印。
                public IntPtr ウィンドウハンドル = IntPtr.Zero;
                public float 視野角deg { get; set; } = 45f;
                public SharpDX.Matrix ビュー変換行列
@@ -59,6 +59,10 @@ namespace FDK.メディア
                public SharpDX.Matrix3x2 拡大行列DPXtoPX => SharpDX.Matrix3x2.Scaling( this.拡大率DPXtoPX横方向, this.拡大率DPXtoPX縦方向 );
                public SharpDX.Matrix3x2 拡大行列PXtoDPX => SharpDX.Matrix3x2.Scaling( this.拡大率PXtoDPX横方向, this.拡大率PXtoDPX縦方向 );
 
+               public デバイスリソース( SharpDX.Size2F 設計画面サイズdpx )
+               {
+                       this.設計画面サイズdpx = 設計画面サイズdpx;
+               }
                public void すべてのリソースを作成する( System.Drawing.Size バックバッファサイズpx, IntPtr ウィンドウハンドル )
                {
                        try
@@ -452,7 +456,7 @@ namespace FDK.メディア
                                        //FDK.Utilities.解放する( ref this.bs_SwapChain ); → スワップチェーンは解放しない(これの生成・解放はデバイスとセットで行う)。
 
                                        // (0,0)は、サイズ依存リソース無効の印。
-                                       this.物理画面サイズpx = new SharpDX.Size2F( 0, 0 );
+                                       this.物理画面サイズpx = new SharpDX.Size2F( 0f, 0f );
                                }
                        }
                        finally
index 21049df..62761a3 100644 (file)
@@ -128,7 +128,7 @@ namespace FDK.メディア
                        }
 
                        if( ( 0.0f >= this.レイアウトサイズdpx.Width ) || ( 0.0f >= this.レイアウトサイズdpx.Height ) )
-                               this.レイアウトサイズdpx = dr.設計画面サイズdpx;
+                               this.レイアウトサイズdpx = new SharpDX.Size2F( dr.設計画面サイズdpx.Width, dr.設計画面サイズdpx.Height );
 
                        this.テキストレイアウト?.Dispose();
                        this.テキストレイアウト = new SharpDX.DirectWrite.TextLayout(
index a5a6f9e..8af7d25 100644 (file)
@@ -11,7 +11,7 @@ namespace SST
                static void Main( string[] args )
                {
                        var sst = new StrokeStyleT();
-                       sst.Run( args );
+                       sst.Run();
                }
        }
 }
index 5d3c051..886d384 100644 (file)
@@ -6,12 +6,11 @@ using System.IO;
 using System.IO.Pipes;
 using System.Linq;
 using System.Windows.Forms;
-using Microsoft.VisualBasic.ApplicationServices;
 using FDK;     // for string 拡張
 
 namespace SST
 {
-       class StrokeStyleT : FDK.ApplicationBase
+       class StrokeStyleT
        {
                // グローバルリソース (static) 
                public static SST.フォルダ フォルダ => ( StrokeStyleT.bs_フォルダ );
@@ -25,7 +24,6 @@ namespace SST
                public static SST.設定.Config Config => ( StrokeStyleT.bs_Config );
                public static bool ビュアーモードである { get; set; } = false;
                public static ConcurrentQueue<SST.ステージ.演奏.ビュアーメッセージ> ビュアーメッセージキュー => ( StrokeStyleT.bs_ビュアーメッセージキュー );
-
                public static void すべての入力デバイスをポーリングする()
                {
                        // hack: 追加の入力デバイスができたら、ここにポーリングコードを追加すること。
@@ -33,7 +31,6 @@ namespace SST
                        StrokeStyleT.MIDI入力?.ポーリングする();
                }
 
-               // get only static property の初期化。
                static StrokeStyleT()
                {
                        // フォルダ変数を真っ先に登録する。(ほかのメンバのコンストラクタでフォルダ変数を利用できるようにするため。)
@@ -41,190 +38,276 @@ namespace SST
                        SST.フォルダ.フォルダ変数を追加する( "Static", StrokeStyleT.フォルダ.StaticFolder );
                        SST.フォルダ.フォルダ変数を追加する( "AppData", StrokeStyleT.フォルダ.AppDataFolder );
                        SST.フォルダ.フォルダ変数を追加する( "User", null );
+               }
+               public StrokeStyleT()
+               {
+                       this.State = ApplicationState.起動;
+                       this.MainForm = new SharpDX.Windows.RenderForm();
+                       this.MainForm.AllowUserResizing = false;        // ユーザはフォームサイズを変更できない。
+                       this.MainForm.UserResized += フォームサイズが変更された;
+               }
 
-                       // その他の static の生成。
-                       StrokeStyleT.bs_ユーザ管理 = new ユーザ.ユーザ管理();
-                       StrokeStyleT.bs_曲ツリー管理 = new 曲.曲ツリー管理();
+               public void Run()
+               {
+                       Debug.Assert( null != this.MainForm );
+                       SharpDX.Windows.RenderLoop.Run( this.MainForm, () => {
+
+                               // アプリケーションの状態に応じて処理分岐。
+                               switch( this.State )
+                               {
+                                       case ApplicationState.起動:
+                                               this.初期化する();
+                                               this.State = ApplicationState.進行描画;
+                                               break;
+
+                                       case ApplicationState.進行描画:
+                                               this.進行描画する();
+                                               if( this.State == ApplicationState.終了 )
+                                                       this.終了する();
+                                               break;
+
+                                       case ApplicationState.終了:
+                                               // 何もしない
+                                               break;
+                               }
+
+                       } );
                }
 
-               protected override void 初期化する()
+               protected SharpDX.Windows.RenderForm MainForm = null;
+               protected enum ApplicationState
                {
+                       起動,
+                       進行描画,
+                       終了,
+               }
+               protected ApplicationState State;
+               protected SharpDX.Size2F 設計画面サイズdpx = SharpDX.Size2F.Empty;
+               protected FDK.メディア.デバイスリソース デバイスリソース = null;
+               protected SST.ステージ.ステージ 最初のダミーステージ = null;
+               protected SST.ステージ.起動.起動ステージ 起動ステージ = null;
+               protected SST.ステージ.タイトル.タイトルステージ タイトルステージ = null;
+               protected SST.ステージ.ログイン.ログインステージ ログインステージ = null;
+               protected SST.ステージ.選曲.選曲ステージ 選曲ステージ = null;
+               protected SST.ステージ.曲読込.曲読込ステージ 曲読込ステージ = null;
+               protected SST.ステージ.演奏.演奏ステージ 演奏ステージ = null;
+               protected SST.ステージ.結果.結果ステージ 結果ステージ = null;
+               protected SST.ステージ.ステージ 現在のステージ = null;
+
+               protected void 初期化する()
+               {
+                       FDK.Log.現在のスレッドに名前をつける( "Render" );
                        FDK.Log.BeginInfo( $"{FDK.Utilities.現在のメソッド名}" );
+                       Debug.Assert( null == this.デバイスリソース, "デバイスリソースの作成前であること。" );
 
-                       lock( this.スレッド間同期 )
-                       {
-                               Debug.Assert( null == this.デバイスリソース, "デバイスリソースの作成前であること。" );
-
-                               this.コマンドライン引数を解析する( Environment.GetCommandLineArgs().Skip( 1 ) );   // 最初の要素は exe ファイル名なのでスキップする。
-
-                               this.MainForm.Text = $"StrokeStyle<T> {System.Reflection.Assembly.GetExecutingAssembly().GetName().Version.ToString()}";
-                               if( StrokeStyleT.ビュアーモードである )
-                                       this.MainForm.Text += " (Viewer)";
-                               this.設計画面サイズdpx = new SharpDX.Size2F( 1920, 1080 );     // 設計画面サイズdpx(固定)
-
-                               #region " コンフィグ を初期化する。"
-                               //----------------
-                               FDK.Log.Info( "コンフィグを初期化します。" );
-                               StrokeStyleT.bs_Config = new 設定.Config();
-                               StrokeStyleT.bs_Config.ConfigXmlを読み込む();
-                               //----------------
-                               #endregion
-                               #region " コンフィグで指定されたウィンドウサイズに変更する。"
-                               //----------------
-                               this.MainForm.ClientSize = new System.Drawing.Size( StrokeStyleT.Config.物理画面サイズpx.Width, StrokeStyleT.Config.物理画面サイズpx.Height );
-                               //----------------
-                               #endregion
-
-                               #region " System.Stopwatch が高解像度タイマを使わないならエラー。"
-                               //-----------------
-                               if( false == System.Diagnostics.Stopwatch.IsHighResolution )
-                                       throw new SSTException( "このシステムは、高解像度タイマをサポートしていません。" );
-                               //-----------------
-                               #endregion
-                               #region " MediaFoundation を起動する。"
-                               //-----------------
-                               SharpDX.MediaFoundation.MediaManager.Startup();
-                               //-----------------
-                               #endregion
-                               #region " Sleep 精度を上げる。"
-                               //-----------------
-                               StrokeStyleT.timeBeginPeriod( 1 );
-                               //-----------------
-                               #endregion
-
-                               #region " ステージを生成する。"
-                               //----------------
-                               this.最初のダミーステージ = new ステージ.ステージ();
-                               this.起動ステージ = new ステージ.起動.起動ステージ();
-                               this.タイトルステージ = new ステージ.タイトル.タイトルステージ();
-                               this.ログインステージ = new ステージ.ログイン.ログインステージ();
-                               this.選曲ステージ = new ステージ.選曲.選曲ステージ();
-                               this.曲読込ステージ = new ステージ.曲読込.曲読込ステージ();
-                               this.演奏ステージ = new ステージ.演奏.演奏ステージ();
-                               this.結果ステージ = new ステージ.結果.結果ステージ();
-                               //----------------
-                               #endregion
-                               #region " ステージのActionを接続する。"
-                               //----------------
-                               this.曲読込ステージ.読込曲のファイルパスを取得する = () => ( ( StrokeStyleT.曲ツリー管理.現在選択されているノード as SST.曲.MusicNode )?.sstfファイルパス );
-                               this.結果ステージ.演奏ステージインスタンスを取得する = () => ( this.演奏ステージ );
-                               this.結果ステージ.BGMを終了する = () => { this.演奏ステージ.BGMを解放する(); };
-                               //----------------
-                               #endregion
-                               #region " ユーザを初期化する。"
-                               //-----------------
-                               FDK.Log.Info( "ユーザ情報を初期化します。" );
-                               StrokeStyleT.ユーザ管理.UsersXmlを読み込む();
-
-                               // ユーザ別の初期化。
-                               foreach( var ユーザ in StrokeStyleT.ユーザ管理.ユーザリスト )
-                                       ユーザ.SourcesXmlを読み込む();
-                               //-----------------
-                               #endregion
-                               #region " WASAPI デバイスを初期化する。"
-                               //----------------
-                               StrokeStyleT.bs_Wasapiデバイス = new FDK.メディア.サウンド.WASAPI排他.ExclusiveDevice();
-                               StrokeStyleT.bs_Wasapiデバイス.初期化する( 15.0f );
-                               //----------------
-                               #endregion
-                               #region " キーボード入力 を初期化する。"
-                               //-----------------
-                               FDK.Log.Info( "キーボード入力デバイスを初期化します。" );
-                               StrokeStyleT.bs_キーボード入力 = new FDK.入力.Keyboard( this.MainForm.Handle );
-                               //-----------------
-                               #endregion
-                               #region " MIDI入力 を初期化する。"
-                               //-----------------
-                               FDK.Log.Info( "MIDI入力デバイスを初期化します。" );
-                               StrokeStyleT.bs_MIDI入力 = new FDK.入力.MidiIn();
-                               //-----------------
-                               #endregion
-
-                               FDK.Log.Info( "最初のダミーステージを開始します。" );
-                               this.現在のステージ = this.最初のダミーステージ;
-
-                               //#warning 全画面モード切替えを KeyDown で仮実装。
-                               this.MainForm.KeyDown += ( target, arg ) => {
-
-                                       // Alt+Enter → 画面モードの切り替え
-                                       if( ( arg.KeyCode == System.Windows.Forms.Keys.Return ) && ( arg.Modifiers == Keys.Alt ) )
-                                       {
-                                               this.全画面モードとウィンドウモードを切り替える();
-                                               arg.Handled = true;
-                                       }
-                               };
-                       }
+                       StrokeStyleT.bs_ユーザ管理 = new ユーザ.ユーザ管理();
+                       StrokeStyleT.bs_曲ツリー管理 = new 曲.曲ツリー管理();
+
+                       #region " コマンドライン引数を解析する。"
+                       //----------------
+                       // 最初の要素は exe ファイル名なのでスキップする。
+                       this.コマンドライン引数を解析する( Environment.GetCommandLineArgs().Skip( 1 ) );
+                       //----------------
+                       #endregion
+
+                       #region " 高解像度タイマを使えないならエラー。"
+                       //-----------------
+                       if( false == System.Diagnostics.Stopwatch.IsHighResolution )
+                               throw new SSTException( "このシステムは、高解像度タイマをサポートしていません。" );
+                       //-----------------
+                       #endregion
+                       #region " MediaFoundation を起動する。"
+                       //-----------------
+                       SharpDX.MediaFoundation.MediaManager.Startup();
+                       //-----------------
+                       #endregion
+                       #region " Sleep 精度を上げる。"
+                       //-----------------
+                       Win32.timeBeginPeriod( 1 );
+                       //-----------------
+                       #endregion
+
+                       #region " コンフィグ を初期化する。"
+                       //----------------
+                       FDK.Log.Info( "コンフィグを初期化します。" );
+                       StrokeStyleT.bs_Config = new 設定.Config();
+                       StrokeStyleT.bs_Config.ConfigXmlを読み込む();
+                       //----------------
+                       #endregion
+
+                       #region " コンフィグで指定されたウィンドウサイズに変更する。"
+                       //----------------
+                       this.MainForm.ClientSize = new System.Drawing.Size( StrokeStyleT.Config.物理画面サイズpx.Width, StrokeStyleT.Config.物理画面サイズpx.Height );
+                       //----------------
+                       #endregion
+                       #region " フォームタイトルと設計画面サイズを設定する。"
+                       //----------------
+                       this.MainForm.Text = $"StrokeStyle<T> {System.Reflection.Assembly.GetExecutingAssembly().GetName().Version.ToString()}";
+                       if( StrokeStyleT.ビュアーモードである )
+                               this.MainForm.Text += " (Viewer)";
+
+                       this.設計画面サイズdpx = new SharpDX.Size2F( 1920f, 1080f );
+                       //----------------
+                       #endregion
+
+                       #region " デバイスリソースを作成する。"
+                       //----------------
+                       FDK.Log.BeginInfo( "デバイスリソースを作成します。" );
+                       FDK.Log.Info( $"設計画面サイズ: {this.設計画面サイズdpx}" );
+                       FDK.Log.Info( $"物理画面サイズ: {this.MainForm.ClientSize}" );
+                       this.デバイスリソース = new FDK.メディア.デバイスリソース( this.設計画面サイズdpx );
+                       this.デバイスリソース.すべてのリソースを作成する( this.MainForm.ClientSize, this.MainForm.Handle );
+                       //----------------
+                       #endregion
+
+                       #region " ステージを生成する。"
+                       //----------------
+                       this.最初のダミーステージ = new ステージ.ステージ();
+                       this.起動ステージ = new ステージ.起動.起動ステージ();
+                       this.タイトルステージ = new ステージ.タイトル.タイトルステージ();
+                       this.ログインステージ = new ステージ.ログイン.ログインステージ();
+                       this.選曲ステージ = new ステージ.選曲.選曲ステージ();
+                       this.曲読込ステージ = new ステージ.曲読込.曲読込ステージ();
+                       this.演奏ステージ = new ステージ.演奏.演奏ステージ();
+                       this.結果ステージ = new ステージ.結果.結果ステージ();
+
+                       // 外部依存アクションを接続する。
+                       this.曲読込ステージ.読込曲のファイルパスを取得する = () => ( ( StrokeStyleT.曲ツリー管理.現在選択されているノード as SST.曲.MusicNode )?.sstfファイルパス );
+                       this.結果ステージ.演奏ステージインスタンスを取得する = () => ( this.演奏ステージ );
+                       this.結果ステージ.BGMを終了する = () => { this.演奏ステージ.BGMを解放する(); };
+                       //----------------
+                       #endregion
+                       #region " ユーザを初期化する。"
+                       //-----------------
+                       FDK.Log.Info( "ユーザ情報を初期化します。" );
+                       StrokeStyleT.ユーザ管理.UsersXmlを読み込む();
+
+                       // ユーザ別の初期化。
+                       foreach( var ユーザ in StrokeStyleT.ユーザ管理.ユーザリスト )
+                               ユーザ.SourcesXmlを読み込む();
+                       //-----------------
+                       #endregion
+                       #region " WASAPI デバイスを初期化する。"
+                       //----------------
+                       StrokeStyleT.bs_Wasapiデバイス = new FDK.メディア.サウンド.WASAPI排他.ExclusiveDevice();
+                       StrokeStyleT.bs_Wasapiデバイス.初期化する( 15.0f );
+                       //----------------
+                       #endregion
+                       #region " キーボード入力 を初期化する。"
+                       //-----------------
+                       FDK.Log.Info( "キーボード入力デバイスを初期化します。" );
+                       StrokeStyleT.bs_キーボード入力 = new FDK.入力.Keyboard( this.MainForm.Handle );
+                       //-----------------
+                       #endregion
+                       #region " MIDI入力 を初期化する。"
+                       //-----------------
+                       FDK.Log.Info( "MIDI入力デバイスを初期化します。" );
+                       StrokeStyleT.bs_MIDI入力 = new FDK.入力.MidiIn();
+                       //-----------------
+                       #endregion
+
+                       FDK.Log.Info( "最初のダミーステージを開始します。" );
+                       this.現在のステージ = this.最初のダミーステージ;
+
+                       //#warning 全画面モード切替えを KeyDown で仮実装。
+                       this.MainForm.KeyDown += ( target, arg ) => {
+
+                               // F11 → 画面モードの切り替え
+                               if( arg.KeyCode == System.Windows.Forms.Keys.F11 )
+                               {
+                                       this.全画面モードとウィンドウモードを切り替える();
+                                       arg.Handled = true;
+                               }
+                       };
 
+                       Debug.Assert( SharpDX.Size2F.Empty != this.設計画面サイズdpx, "初期化メソッド内で設計画面サイズを設定してあること。" );
                        FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
                }
-               protected override void 終了する()
+               protected void 終了する()
                {
                        FDK.Log.BeginInfo( $"{FDK.Utilities.現在のメソッド名}" );
-
-                       lock( this.スレッド間同期 )
-                       {
-                               Debug.Assert( null != this.デバイスリソース, "デバイスリソースが解放される前であること。" );
-
-                               #region " ステージを終了し、解放する。"
-                               //----------------
-                               if( ( null != this.現在のステージ ) && ( this.現在のステージ.活性化している ) )            // 念のため
-                                       this.現在のステージ.非活性化する( this.デバイスリソース );
-
-                               this.最初のダミーステージ = null;
-                               this.起動ステージ = null;
-                               this.タイトルステージ = null;
-                               this.ログインステージ = null;
-                               this.選曲ステージ = null;
-                               this.曲読込ステージ = null;
-                               this.演奏ステージ = null;
-                               this.結果ステージ = null;
-                               //----------------
-                               #endregion
-                               #region " MIDI入力 を解放する。"
-                               //-----------------
-                               FDK.Log.Info( "MIDI入力デバイスを解放します。" );
-                               FDK.Utilities.解放する( ref StrokeStyleT.bs_MIDI入力 );
-                               //-----------------
-                               #endregion
-                               #region " キーボード入力 を解放する。"
-                               //-----------------
-                               FDK.Log.Info( "キーボード入力デバイスを解放します。" );
-                               FDK.Utilities.解放する( ref StrokeStyleT.bs_キーボード入力 );
-                               //-----------------
-                               #endregion
-                               #region " WASAPIデバイスを解放する。"
-                               //----------------
-                               FDK.Log.Info( "WASAPIデバイスを解放します。" );
-                               FDK.Utilities.解放する( ref StrokeStyleT.bs_Wasapiデバイス );
-                               //----------------
-                               #endregion
-                               #region " コンフィグを解放する。"
-                               //----------------
-                               FDK.Log.Info( "コンフィグを解放します。" );
-                               StrokeStyleT.bs_Config.ConfigXmlを保存する();
-                               StrokeStyleT.bs_Config = null;
-                               //----------------
-                               #endregion
-                               #region " MediaFoundation を終了する。"
-                               //-----------------
-                               SharpDX.MediaFoundation.MediaManager.Shutdown();
-                               //-----------------
-                               #endregion
-                       }
+                       Debug.Assert( null != this.デバイスリソース, "デバイスリソースが解放される前であること。" );
+
+                       #region " ステージを終了し、解放する。"
+                       //----------------
+                       if( ( null != this.現在のステージ ) && ( this.現在のステージ.活性化している ) )      // 念のため
+                               this.現在のステージ.非活性化する( this.デバイスリソース );
+
+                       this.最初のダミーステージ = null;
+                       this.起動ステージ = null;
+                       this.タイトルステージ = null;
+                       this.ログインステージ = null;
+                       this.選曲ステージ = null;
+                       this.曲読込ステージ = null;
+                       this.演奏ステージ = null;
+                       this.結果ステージ = null;
+                       //----------------
+                       #endregion
+
+                       #region " デバイスリソースを解放する。"
+                       //----------------
+                       FDK.Utilities.解放する( ref this.デバイスリソース );
+                       //----------------
+                       #endregion
+
+                       #region " MIDI入力デバイスを解放する。"
+                       //-----------------
+                       FDK.Log.Info( "MIDI入力デバイスを解放します。" );
+                       FDK.Utilities.解放する( ref StrokeStyleT.bs_MIDI入力 );
+                       //-----------------
+                       #endregion
+                       #region " キーボード入力デバイスを解放する。"
+                       //-----------------
+                       FDK.Log.Info( "キーボード入力デバイスを解放します。" );
+                       FDK.Utilities.解放する( ref StrokeStyleT.bs_キーボード入力 );
+                       //-----------------
+                       #endregion
+                       #region " WASAPIデバイスを解放する。"
+                       //----------------
+                       FDK.Log.Info( "WASAPIデバイスを解放します。" );
+                       FDK.Utilities.解放する( ref StrokeStyleT.bs_Wasapiデバイス );
+                       //----------------
+                       #endregion
+                       
+                       #region " コンフィグを保存し、解放する。"
+                       //----------------
+                       FDK.Log.Info( "コンフィグを解放します。" );
+                       StrokeStyleT.bs_Config.ConfigXmlを保存する();
+                       StrokeStyleT.bs_Config = null;
+                       //----------------
+                       #endregion
+                       #region " フォームを閉じる。"
+                       //----------------
+                       FDK.Utilities.解放する( ref this.MainForm );
+                       //----------------
+                       #endregion
+
+                       #region " MediaFoundation を終了する。"
+                       //-----------------
+                       SharpDX.MediaFoundation.MediaManager.Shutdown();
+                       //-----------------
+                       #endregion
 
                        FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
                }
-               protected override void シーンを描画する()
+               protected void 進行描画する()
                {
-                       // このメソッドは、GUIスレッドではなく進行描画スレッドから呼び出されるので注意。(FDK.ApplicationBase.進行描画スレッド処理() を参照。)
-
-                       var swapChain = (SharpDX.DXGI.SwapChain1) null;
+                       #region " D3Dデバイスが消失していれば再構築する。"
+                       //----------------
+                       bool 異常発生 = false;
+                       this.デバイスリソース.D3Dデバイスが消失していれば再構築する( out 異常発生 );
 
-                       // 現在のステージを進行描画する。
-                       lock( this.スレッド間同期 )
+                       // 再構築不可能な異常の場合は、即終了する。
+                       if( 異常発生 )
+                       {
+                               this.State = ApplicationState.終了;
+                               return;
+                       }
+                       //----------------
+                       #endregion
+                       #region " 描画の前処理を行う。"
+                       //----------------
                        {
-                               #region " 描画の準備を行う。"
-                               //----------------
                                var d3dDevice = (SharpDX.Direct3D11.Device) null;
                                using( var d3dLock = new FDK.同期.AutoD3DDeviceLock( this.デバイスリソース.DXGIDeviceManager, out d3dDevice ) )
                                using( d3dDevice )
@@ -240,326 +323,356 @@ namespace SST
                                                depth: 1.0f,
                                                stencil: 0 );
                                }
-                               //----------------
-                               #endregion
-
-                               this.現在のステージ?.進行描画する( this.デバイスリソース );
-
-                               if( StrokeStyleT.Config.垂直帰線待ちを行う )
+                       }
+                       //----------------
+                       #endregion
+                       #region " 現在のステージを進行描画する。"
+                       //----------------
+                       this.現在のステージ?.進行描画する( this.デバイスリソース );
+                       //----------------
+                       #endregion
+                       #region " スワップチェーンを表示する。"
+                       //----------------
+                       if( StrokeStyleT.Config.垂直帰線待ちを行う )
+                       {
+                               // We recommend that you use Flush when the CPU waits for an arbitrary amount of time
+                               // ( such as when you call the Sleep function). 
+                               // https://msdn.microsoft.com/ja-jp/library/windows/desktop/ff476425(v=vs.85).aspx
+                               var d3dDevice = (SharpDX.Direct3D11.Device) null;
+                               using( var d3dLock = new FDK.同期.AutoD3DDeviceLock( this.デバイスリソース.DXGIDeviceManager, out d3dDevice ) )
+                               using( d3dDevice )
+                               using( var d3dContext = d3dDevice.ImmediateContext )
                                {
-                                       #region " Flush する。"
-                                       //----------------
-                                       d3dDevice = (SharpDX.Direct3D11.Device) null;
-                                       using( var d3dLock = new FDK.同期.AutoD3DDeviceLock( this.デバイスリソース.DXGIDeviceManager, out d3dDevice ) )
-                                       using( d3dDevice )
-                                       using( var d3dContext = d3dDevice.ImmediateContext )
-                                       {
-                                               // We recommend that you use Flush when the CPU waits for an arbitrary amount of time
-                                               // ( such as when you call the Sleep function). 
-                                               // https://msdn.microsoft.com/ja-jp/library/windows/desktop/ff476425(v=vs.85).aspx
-                                               d3dContext.Flush();
-                                       }
-                                       //----------------
-                                       #endregion
+                                       d3dContext.Flush();
                                }
-
-                               swapChain = this.デバイスリソース.SwapChain1;
                        }
 
-                       // スワップチェーンを表示する。垂直帰線待ちなどで時間がかかるので、この部分はスレッド排他領域の外に配置すること。
-                       swapChain.Present(
+                       this.デバイスリソース.SwapChain1.Present(
                                ( StrokeStyleT.Config.垂直帰線待ちを行う ) ? 1 : 0,
                                SharpDX.DXGI.PresentFlags.None );
-
-                       // ステージの状態をチェックし、必要あれば遷移またはアプリを終了する。
-                       bool アプリを終了せよ = false;
-                       lock( this.スレッド間同期)
+                       //----------------
+                       #endregion
+                       #region " ステージの状態をチェックし、必要あれば遷移またはアプリを終了する。"
+                       //----------------
+                       if( null != this.現在のステージ )
                        {
-                               if( null != this.現在のステージ )
+                               switch( this.現在のステージ.GetType().Name )
                                {
-                                       switch( this.現在のステージ.GetType().Name )
-                                       {
-                                               case nameof( ステージ.ステージ ):
-                                                       #region " ビュアーモード → AutoPlayerでログインして曲読込ステージへ。"
-                                                       //----------------
-                                                       if( StrokeStyleT.ビュアーモードである )
-                                                       {
-                                                               FDK.Log.Info( "ビュアーモード: AutoPlayer ユーザでログインします。" );
-                                                               this.ログインする( Properties.Resources.AUTOPLAYER );
+                                       case nameof( ステージ.ステージ ):
+                                               #region " ビュアーモード → AutoPlayerでログインして曲読込ステージへ。"
+                                               //----------------
+                                               if( StrokeStyleT.ビュアーモードである )
+                                               {
+                                                       FDK.Log.Info( "ビュアーモード: AutoPlayer ユーザでログインします。" );
+                                                       this.ログインする( Properties.Resources.AUTOPLAYER );
 
-                                                               this.曲読込ステージ.読込曲のファイルパスを取得する = () => ( null );      // 今は null 。あとでメッセージキューを見る。
-                                                               this.曲読込ステージ.活性化する( this.デバイスリソース );
-                                                               this.現在のステージ = this.曲読込ステージ;
-                                                       }
-                                                       //----------------
-                                                       #endregion
-                                                       #region " 通常モード → 起動ステージへ。"
-                                                       //----------------
-                                                       else
-                                                       {
-                                                               this.起動ステージ.活性化する( this.デバイスリソース );
-                                                               this.現在のステージ = this.起動ステージ;
-                                                       }
-                                                       //----------------
-                                                       #endregion
-                                                       break;
-
-                                               case nameof( ステージ.起動.起動ステージ ):
-                                                       #region " 終了 → タイトルステージへ。"
-                                                       //---------------
-                                                       if( this.起動ステージ.現在のフェーズ == ステージ.起動.起動ステージ.フェーズ.終了 )
-                                                       {
-                                                               this.現在のステージ.非活性化する( this.デバイスリソース );
-                                                               this.現在のステージ = this.タイトルステージ;
-                                                               this.現在のステージ.活性化する( this.デバイスリソース );
-                                                       }
-                                                       //---------------
-                                                       #endregion
-                                                       break;
-
-                                               case nameof( ステージ.タイトル.タイトルステージ ):
-                                                       #region " 確定 → ログインステージへ。"
-                                                       //---------------
-                                                       if( this.タイトルステージ.現在のフェーズ == ステージ.タイトル.タイトルステージ.フェーズ.確定 )
-                                                       {
-                                                               this.現在のステージ.非活性化する( this.デバイスリソース );
-                                                               this.現在のステージ = this.ログインステージ;
-                                                               this.現在のステージ.活性化する( this.デバイスリソース );
-                                                       }
-                                                       //---------------
-                                                       #endregion
-                                                       #region " キャンセル → アプリを終了する。"
-                                                       //---------------
-                                                       else if( this.タイトルステージ.現在のフェーズ == ステージ.タイトル.タイトルステージ.フェーズ.キャンセル )
+                                                       this.曲読込ステージ.読込曲のファイルパスを取得する = () => ( null );  // 今は null 。あとでメッセージキューを見る。
+                                                       this.曲読込ステージ.活性化する( this.デバイスリソース );
+                                                       this.現在のステージ = this.曲読込ステージ;
+                                               }
+                                               //----------------
+                                               #endregion
+                                               #region " 通常モード → 起動ステージへ。"
+                                               //----------------
+                                               else
+                                               {
+                                                       this.起動ステージ.活性化する( this.デバイスリソース );
+                                                       this.現在のステージ = this.起動ステージ;
+                                               }
+                                               //----------------
+                                               #endregion
+                                               break;
+
+                                       case nameof( ステージ.起動.起動ステージ ):
+                                               #region " 終了 → タイトルステージへ。"
+                                               //---------------
+                                               if( this.起動ステージ.現在のフェーズ == ステージ.起動.起動ステージ.フェーズ.終了 )
+                                               {
+                                                       this.現在のステージ.非活性化する( this.デバイスリソース );
+                                                       this.現在のステージ = this.タイトルステージ;
+                                                       this.現在のステージ.活性化する( this.デバイスリソース );
+                                               }
+                                               //---------------
+                                               #endregion
+                                               break;
+
+                                       case nameof( ステージ.タイトル.タイトルステージ ):
+                                               #region " 確定 → ログインステージへ。"
+                                               //---------------
+                                               if( this.タイトルステージ.現在のフェーズ == ステージ.タイトル.タイトルステージ.フェーズ.確定 )
+                                               {
+                                                       this.現在のステージ.非活性化する( this.デバイスリソース );
+                                                       this.現在のステージ = this.ログインステージ;
+                                                       this.現在のステージ.活性化する( this.デバイスリソース );
+                                               }
+                                               //---------------
+                                               #endregion
+                                               #region " キャンセル → アプリを終了する。"
+                                               //---------------
+                                               else if( this.タイトルステージ.現在のフェーズ == ステージ.タイトル.タイトルステージ.フェーズ.キャンセル )
+                                               {
+                                                       this.現在のステージ.非活性化する( this.デバイスリソース );
+                                                       this.現在のステージ = null;
+                                                       this.State = ApplicationState.終了;
+                                               }
+                                               //---------------
+                                               #endregion
+                                               break;
+
+                                       case nameof( ステージ.ログイン.ログインステージ ):
+                                               #region " 確定 → ログイン処理を行って、選曲ステージへ。"
+                                               //---------------
+                                               if( this.ログインステージ.現在のフェーズ == ステージ.ログイン.ログインステージ.フェーズ.確定 )
+                                               {
+                                                       var user = StrokeStyleT.ユーザ管理.現在選択されているユーザ;
+
+                                                       if( null != user )
                                                        {
-                                                               this.現在のステージ.非活性化する( this.デバイスリソース );
-                                                               this.現在のステージ = null;
-                                                               アプリを終了せよ = true;
+                                                               foreach( var path in user.曲の検索元フォルダパスのリスト )
+                                                                       SST.曲.曲ツリー管理.フォルダから曲を再帰的に検索して子ノードリストに追加する( user.曲ツリーのルートノード, path );
+
+                                                               StrokeStyleT.曲ツリー管理.現在の管理対象ツリー = StrokeStyleT.ユーザ管理.現在選択されているユーザ.曲ツリーのルートノード;
                                                        }
-                                                       //---------------
-                                                       #endregion
-                                                       break;
-
-                                               case nameof( ステージ.ログイン.ログインステージ ):
-                                                       #region " 確定 → ログイン処理を行って、選曲ステージへ。"
-                                                       //---------------
-                                                       if( this.ログインステージ.現在のフェーズ == ステージ.ログイン.ログインステージ.フェーズ.確定 )
-                                                       {
-                                                               var user = StrokeStyleT.ユーザ管理.現在選択されているユーザ;
 
-                                                               if( null != user )
-                                                               {
-                                                                       foreach( var path in user.曲の検索元フォルダパスのリスト )
-                                                                               SST.曲.曲ツリー管理.フォルダから曲を再帰的に検索して子ノードリストに追加する( user.曲ツリーのルートノード, path );
+                                                       this.現在のステージ.非活性化する( this.デバイスリソース );
+                                                       this.現在のステージ = this.選曲ステージ;
+                                                       this.現在のステージ.活性化する( this.デバイスリソース );
+                                               }
+                                               //---------------
+                                               #endregion
+                                               #region " キャンセル → タイトルステージへ。"
+                                               //---------------
+                                               else if( this.ログインステージ.現在のフェーズ == ステージ.ログイン.ログインステージ.フェーズ.キャンセル )
+                                               {
+                                                       this.現在のステージ.非活性化する( this.デバイスリソース );
+                                                       this.現在のステージ = this.タイトルステージ;
+                                                       this.現在のステージ.活性化する( this.デバイスリソース );
+                                               }
+                                               //---------------
+                                               #endregion
+                                               break;
+
+                                       case nameof( ステージ.選曲.選曲ステージ ):
+                                               #region " 曲確定 → 曲読込ステージへ。"
+                                               //---------------
+                                               if( this.選曲ステージ.現在のフェーズ == ステージ.選曲.選曲ステージ.フェーズ.曲確定 )
+                                               {
+                                                       // 曲ノードが選択されていることを確認。
+                                                       Trace.Assert( null != StrokeStyleT.曲ツリー管理.現在選択されているノード, "[バグあり] 選択曲が null です。" );
+                                                       this.現在のステージ.非活性化する( this.デバイスリソース );
+                                                       this.現在のステージ = this.曲読込ステージ;
+                                                       this.現在のステージ.活性化する( this.デバイスリソース );
+                                               }
+                                               //---------------
+                                               #endregion
+                                               #region " キャンセル → アプリを終了する。"
+                                               //---------------
+                                               else if( this.選曲ステージ.現在のフェーズ == ステージ.選曲.選曲ステージ.フェーズ.キャンセル )
+                                               {
+                                                       this.現在のステージ.非活性化する( this.デバイスリソース );
+                                                       this.現在のステージ = null;
+                                                       this.State = ApplicationState.終了;
+                                               }
+                                               //---------------
+                                               #endregion
+                                               break;
+
+                                       case nameof( ステージ.曲読込.曲読込ステージ ):
+                                               #region " 終了 → 演奏ステージへ。"
+                                               //--------------------
+                                               if( this.曲読込ステージ.現在のフェーズ == ステージ.曲読込.曲読込ステージ.フェーズ.終了 )
+                                               {
+                                                       this.現在のステージ.非活性化する( this.デバイスリソース );
 
-                                                                       StrokeStyleT.曲ツリー管理.現在の管理対象ツリー = StrokeStyleT.ユーザ管理.現在選択されているユーザ.曲ツリーのルートノード;
-                                                               }
+                                                       if( StrokeStyleT.ビュアーモードである )
+                                                               this.デバイス情報を出力する();
 
-                                                               this.現在のステージ.非活性化する( this.デバイスリソース );
-                                                               this.現在のステージ = this.選曲ステージ;
-                                                               this.現在のステージ.活性化する( this.デバイスリソース );
-                                                       }
-                                                       //---------------
-                                                       #endregion
-                                                       #region " キャンセル → タイトルステージへ。"
-                                                       //---------------
-                                                       else if( this.ログインステージ.現在のフェーズ == ステージ.ログイン.ログインステージ.フェーズ.キャンセル )
-                                                       {
-                                                               this.現在のステージ.非活性化する( this.デバイスリソース );
-                                                               this.現在のステージ = this.タイトルステージ;
-                                                               this.現在のステージ.活性化する( this.デバイスリソース );
-                                                       }
-                                                       //---------------
-                                                       #endregion
-                                                       break;
-
-                                               case nameof( ステージ.選曲.選曲ステージ ):
-                                                       #region " 曲確定 → 曲読込ステージへ。"
-                                                       //---------------
-                                                       if( this.選曲ステージ.現在のフェーズ == ステージ.選曲.選曲ステージ.フェーズ.曲確定 )
-                                                       {
-                                                               // 曲ノードが選択されていることを確認。
-                                                               Trace.Assert( null != StrokeStyleT.曲ツリー管理.現在選択されているノード, "[バグあり] 選択曲が null です。" );
-                                                               this.現在のステージ.非活性化する( this.デバイスリソース );
-                                                               this.現在のステージ = this.曲読込ステージ;
-                                                               this.現在のステージ.活性化する( this.デバイスリソース );
-                                                       }
-                                                       //---------------
-                                                       #endregion
-                                                       #region " キャンセル → アプリを終了する。"
-                                                       //---------------
-                                                       else if( this.選曲ステージ.現在のフェーズ == ステージ.選曲.選曲ステージ.フェーズ.キャンセル )
-                                                       {
-                                                               this.現在のステージ.非活性化する( this.デバイスリソース );
-                                                               this.現在のステージ = null;
-                                                               アプリを終了せよ = true;
-                                                       }
-                                                       //---------------
-                                                       #endregion
-                                                       break;
-
-                                               case nameof( ステージ.曲読込.曲読込ステージ ):
-                                                       #region " 終了 → 演奏ステージへ。"
-                                                       //--------------------
-                                                       if( this.曲読込ステージ.現在のフェーズ == ステージ.曲読込.曲読込ステージ.フェーズ.終了 )
+                                                       this.現在のステージ = this.演奏ステージ;
+                                                       this.現在のステージ.活性化する( this.デバイスリソース );
+                                               }
+                                               //--------------------
+                                               #endregion
+                                               break;
+
+                                       case nameof( ステージ.演奏.演奏ステージ ):
+                                               #region " 演奏終了 → 結果ステージへ。"
+                                               //--------------------
+                                               if( this.演奏ステージ.現在のフェーズ.Value == ステージ.演奏.演奏ステージ.フェーズ.クリアor失敗 )
+                                               {
+                                                       this.現在のステージ.非活性化する( this.デバイスリソース );
+                                                       this.現在のステージ = this.結果ステージ;
+                                                       this.現在のステージ.活性化する( this.デバイスリソース );
+                                               }
+                                               //--------------------
+                                               #endregion
+                                               #region " キャンセル → 選曲ステージへ。"
+                                               //--------------------
+                                               if( this.演奏ステージ.現在のフェーズ.Value == ステージ.演奏.演奏ステージ.フェーズ.キャンセル )
+                                               {
+                                                       this.現在のステージ.非活性化する( this.デバイスリソース );
+                                                       this.現在のステージ = this.選曲ステージ;
+                                                       this.現在のステージ.活性化する( this.デバイスリソース );
+                                               }
+                                               //--------------------
+                                               #endregion
+                                               #region " ビュアーメッセージ受信 → 曲読込ステージへ。"
+                                               //--------------------
+                                               if( this.演奏ステージ.現在のフェーズ.Value == ステージ.演奏.演奏ステージ.フェーズ.ビュアーメッセージ待機中 )
+                                               {
+                                                       var msg = (SST.ステージ.演奏.ビュアーメッセージ) null;
+                                                       if( ( 0 < StrokeStyleT.ビュアーメッセージキュー.Count ) &&
+                                                               ( StrokeStyleT.ビュアーメッセージキュー.TryDequeue( out msg ) ) )
                                                        {
-                                                               this.現在のステージ.非活性化する( this.デバイスリソース );
+                                                               FDK.Log.Info( "ビュアーメッセージを受信しました。" );
+                                                               this.曲読込ステージ.読込曲のファイルパスを取得する = () => { return msg.曲ファイルパス; };
 
-                                                               if( StrokeStyleT.ビュアーモードである )
-                                                                       this.デバイス情報を出力する();
-
-                                                               this.現在のステージ = this.演奏ステージ;
-                                                               this.現在のステージ.活性化する( this.デバイスリソース );
-                                                       }
-                                                       //--------------------
-                                                       #endregion
-                                                       break;
-
-                                               case nameof( ステージ.演奏.演奏ステージ ):
-                                                       #region " 演奏終了 → 結果ステージへ。"
-                                                       //--------------------
-                                                       if( this.演奏ステージ.現在のフェーズ.Value == ステージ.演奏.演奏ステージ.フェーズ.クリアor失敗 )
-                                                       {
-                                                               this.現在のステージ.非活性化する( this.デバイスリソース );
-                                                               this.現在のステージ = this.結果ステージ;
-                                                               this.現在のステージ.活性化する( this.デバイスリソース );
-                                                       }
-                                                       //--------------------
-                                                       #endregion
-                                                       #region " キャンセル → 選曲ステージへ。"
-                                                       //--------------------
-                                                       if( this.演奏ステージ.現在のフェーズ.Value == ステージ.演奏.演奏ステージ.フェーズ.キャンセル )
-                                                       {
-                                                               this.現在のステージ.非活性化する( this.デバイスリソース );
-                                                               this.現在のステージ = this.選曲ステージ;
-                                                               this.現在のステージ.活性化する( this.デバイスリソース );
-                                                       }
-                                                       //--------------------
-                                                       #endregion
-                                                       #region " ビュアーメッセージ受信 → 曲読込ステージへ。"
-                                                       //--------------------
-                                                       if( this.演奏ステージ.現在のフェーズ.Value == ステージ.演奏.演奏ステージ.フェーズ.ビュアーメッセージ待機中 )
-                                                       {
-                                                               var msg = (SST.ステージ.演奏.ビュアーメッセージ) null;
-                                                               if( ( 0 < StrokeStyleT.ビュアーメッセージキュー.Count ) &&
-                                                                       ( StrokeStyleT.ビュアーメッセージキュー.TryDequeue( out msg ) ) )
-                                                               {
-                                                                       FDK.Log.Info( "ビュアーメッセージを受信しました。" );
-                                                                       this.曲読込ステージ.読込曲のファイルパスを取得する = () => { return msg.曲ファイルパス; };
-
-                                                                       this.現在のステージ.非活性化する( this.デバイスリソース );
-                                                                       this.演奏ステージ.BGMを解放する();
-                                                                       this.現在のステージ = this.曲読込ステージ;
-                                                                       this.現在のステージ.活性化する( this.デバイスリソース );
-                                                               }
-                                                       }
-                                                       //--------------------
-                                                       #endregion
-                                                       break;
-
-                                               case nameof( ステージ.結果.結果ステージ ):
-                                                       #region " 終了 → 選曲ステージへ。"
-                                                       //--------------------
-                                                       if( this.結果ステージ.現在のフェーズ == ステージ.結果.結果ステージ.フェーズ.終了 )
-                                                       {
                                                                this.現在のステージ.非活性化する( this.デバイスリソース );
-                                                               this.現在のステージ = this.選曲ステージ;
+                                                               this.演奏ステージ.BGMを解放する();
+                                                               this.現在のステージ = this.曲読込ステージ;
                                                                this.現在のステージ.活性化する( this.デバイスリソース );
                                                        }
-                                                       //--------------------
-                                                       #endregion
-                                                       break;
-                                       }
+                                               }
+                                               //--------------------
+                                               #endregion
+                                               break;
+
+                                       case nameof( ステージ.結果.結果ステージ ):
+                                               #region " 終了 → 選曲ステージへ。"
+                                               //--------------------
+                                               if( this.結果ステージ.現在のフェーズ == ステージ.結果.結果ステージ.フェーズ.終了 )
+                                               {
+                                                       this.現在のステージ.非活性化する( this.デバイスリソース );
+                                                       this.現在のステージ = this.選曲ステージ;
+                                                       this.現在のステージ.活性化する( this.デバイスリソース );
+                                               }
+                                               //--------------------
+                                               #endregion
+                                               break;
                                }
                        }
+                       //----------------
+                       #endregion
+               }
+               protected void デバイス依存リソースを解放する()
+               {
+                       FDK.Log.BeginInfo( $"{FDK.Utilities.現在のメソッド名}" );
+                       Debug.Assert( null != this.デバイスリソース );  // 解放前であること。
 
-                       if( アプリを終了せよ )
-                       {
-                               // GUIスレッド上で、ウィンドウを閉じる。
-                               this.MainForm.BeginInvoke( new Action( () => { this.MainForm.Close(); } ) );
-                       }
+                       this.現在のステージ?.デバイス依存リソースを解放する( this.デバイスリソース );
+
+                       FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
                }
-               protected override void デバイス依存リソースを解放する()
+               protected void デバイス依存リソースを再構築する()
                {
                        FDK.Log.BeginInfo( $"{FDK.Utilities.現在のメソッド名}" );
+                       Debug.Assert( null != this.デバイスリソース );  // 再生成済みであること。
 
-                       lock( this.スレッド間同期 )
-                       {
-                               Debug.Assert( null != this.デバイスリソース );  // 解放前であること。
-                               this.現在のステージ?.デバイス依存リソースを解放する( this.デバイスリソース );
-                       }
+                       this.現在のステージ?.デバイス依存リソースを作成する( this.デバイスリソース );
 
                        FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
                }
-               protected override void デバイス依存リソースを再構築する()
+
+               private void フォームサイズが変更された( object sender, EventArgs e )
                {
                        FDK.Log.BeginInfo( $"{FDK.Utilities.現在のメソッド名}" );
+                       FDK.Log.Info( $"新しいクライアントサイズ = {this.MainForm.ClientSize}" );
 
-                       lock( this.スレッド間同期 )
+                       #region " 実行条件チェック。"
+                       //----------------
+                       if( null == this.デバイスリソース )
+                       {
+                               FDK.Log.Info( " まだ初期化されてないので、何もしません。" );
+                               FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
+                               return;
+                       }
+                       if( this.MainForm.WindowState == FormWindowState.Minimized )
                        {
-                               Debug.Assert( null != this.デバイスリソース );  // 再生成済みであること。
-                               this.現在のステージ?.デバイス依存リソースを作成する( this.デバイスリソース );
+                               FDK.Log.Info( "最小化されました。" );
+                               FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
+                               return; // 何もしない
                        }
+                       if( this.State != ApplicationState.進行描画 )
+                       {
+                               FDK.Log.Info( " アプリケーションの状態が進行描画じゃないので、何もしません。" );
+                               FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
+                               return;
+                       }
+                       //----------------
+                       #endregion
+
+                       Debug.Assert( null != this.デバイスリソース, "デバイスリソースが作成済みであること。" );
+
+                       // 現在の画面モードを取得しておく。(Alt+TABなど、勝手に全画面を解除されることもあるので。)
+                       SharpDX.Mathematics.Interop.RawBool fullscreen;
+                       SharpDX.DXGI.Output outputTarget;
+                       this.デバイスリソース.SwapChain1.GetFullscreenState( out fullscreen, out outputTarget );
+                       this.MainForm.IsFullscreen = fullscreen;
+                       outputTarget?.Dispose();
+                       FDK.Log.Info( $"現在、全画面モードである = {this.MainForm.IsFullscreen}" );
+
+                       // (1) リソースを解放して、
+                       this.デバイス依存リソースを解放する();
+                       this.デバイスリソース.サイズに依存するリソースを解放する();
+
+                       // (2) 物理画面サイズを変更して、
+                       this.デバイスリソース.物理画面サイズpx = new SharpDX.Size2F( this.MainForm.ClientSize.Width, this.MainForm.ClientSize.Height );
+
+                       // (3) リソースを再構築する。
+                       this.デバイスリソース.サイズに依存するリソースを作成する();
+                       this.デバイス依存リソースを再構築する();
 
                        FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
                }
-               /// <summary>
-               /// アプリが二重起動されたときに発生するイベント。
-               /// </summary>
-               /// <remarks>
-               /// 後続のインスタンスは起動せず、既存のインスタンスに対してこのイベントが発生する。
-               /// eventArg.CommandLine で、後続のインスタンスのコマンドライン引数を確認することができる。
-               /// </remarks>
-               protected override void OnStartupNextInstance( StartupNextInstanceEventArgs eventArgs )
+               private void 全画面モードとウィンドウモードを切り替える()
                {
                        FDK.Log.BeginInfo( $"{FDK.Utilities.現在のメソッド名}" );
 
-                       lock( this.スレッド間同期 )
+                       this.MainForm.IsFullscreen = !this.MainForm.IsFullscreen;
+                       this.デバイスリソース.SwapChain1.SetFullscreenState( this.MainForm.IsFullscreen, null );
+
+                       FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
+               }
+               private void ログインする( string ユーザ名 )
+               {
+                       StrokeStyleT.ユーザ管理.ユーザを選択する( ユーザ名 );
+                       FDK.Log.Info( $"ユーザが選択されました。[{StrokeStyleT.ユーザ管理.現在選択されているユーザ.名前}]" );
+               }
+               private void デバイス情報を出力する()
+               {
+                       using( var stream = new NamedPipeClientStream( "SSTFEditor Viewer Device Information" ) )
                        {
-                               if( StrokeStyleT.ビュアーモードである )
+                               try
                                {
-                                       this.コマンドライン引数を解析する( eventArgs.CommandLine );
+                                       stream.Connect( 1000 );
+
+                                       using( var writer = new StreamWriter( stream ) )
+                                       {
+                                               writer.WriteLine( $"SoundDevice.Delay={StrokeStyleT.Wasapiデバイス.遅延ms.ToString()}" );
+                                               FDK.Log.Info( "デバイス情報を出力しました。" );
+                                       }
                                }
-                               else
+                               catch( TimeoutException )
                                {
-                                       FDK.Log.ERROR( "現在、ビュアーモードではありません。" );
+                                       FDK.Log.WARNING( "SSTFEditor ビュアー用パイプへの接続がタイムアウトしました。SSTFEditor が起動していない可能性があります。" );
                                }
                        }
-
-                       FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
                }
+               private void 二重起動された( string[] args )
+               {
+                       FDK.Log.BeginInfo( $"{FDK.Utilities.現在のメソッド名}" );
 
-               // 各ステージの、唯一のインスタンス。Config 生成後に生成するので、readonlyにはしない。
-               protected SST.ステージ.ステージ 最初のダミーステージ = null;
-               protected SST.ステージ.起動.起動ステージ 起動ステージ = null;
-               protected SST.ステージ.タイトル.タイトルステージ タイトルステージ = null;
-               protected SST.ステージ.ログイン.ログインステージ ログインステージ = null;
-               protected SST.ステージ.選曲.選曲ステージ 選曲ステージ = null;
-               protected SST.ステージ.曲読込.曲読込ステージ 曲読込ステージ = null;
-               protected SST.ステージ.演奏.演奏ステージ 演奏ステージ = null;
-               protected SST.ステージ.結果.結果ステージ 結果ステージ = null;
-
-               private SST.ステージ.ステージ 現在のステージ = null;
-
-               #region " バックストア。"
-               //----------------
-               private static SST.フォルダ bs_フォルダ = null;
-               private static FDK.入力.Keyboard bs_キーボード入力 = null;
-               private static FDK.入力.MidiIn bs_MIDI入力 = null;
-               private static FDK.メディア.サウンド.WASAPI排他.ExclusiveDevice bs_Wasapiデバイス = null;
-               private static readonly Random bs_乱数 = new Random( DateTime.Now.Millisecond );
-               private static SST.ユーザ.ユーザ管理 bs_ユーザ管理 = null;
-               private static SST.曲.曲ツリー管理 bs_曲ツリー管理 = null;
-               private static SST.設定.Config bs_Config = null;
-               private static readonly ConcurrentQueue<SST.ステージ.演奏.ビュアーメッセージ> bs_ビュアーメッセージキュー = new ConcurrentQueue<ステージ.演奏.ビュアーメッセージ>();
-               //----------------
-               #endregion
+                       if( StrokeStyleT.ビュアーモードである )
+                       {
+                               this.コマンドライン引数を解析する( args );
+                       }
+                       else
+                       {
+                               FDK.Log.ERROR( "アプリが二重起動されましたが、先行アプリがビュアーモードではないので何もしません。" );
+                       }
 
-               /// <summary>
-               /// コマンドライン引数を解析して、ビュアーモードの設定があればそれを返す。
-               /// </summary>
+                       FDK.Log.EndInfo( $"{FDK.Utilities.現在のメソッド名}" );
+               }
                /// <param name="args">コマンドライン引数の列挙。exeファイル名は含まない。</param>
-               /// <return>引数を反映したビュアーモード変数。</return>
                private void コマンドライン引数を解析する( IEnumerable<string> args )
                {
                        FDK.Log.BeginInfo( $"{FDK.Utilities.現在のメソッド名}" );
@@ -578,10 +691,10 @@ namespace SST
                                {
                                        // 新しいメッセージを生成する。
                                        var msg = new ステージ.演奏.ビュアーメッセージ() {
-                                               種別 = ステージ.演奏.ビュアーメッセージ.E種別.演奏開始,          // 規定値は「演奏開始」
+                                               種別 = ステージ.演奏.ビュアーメッセージ.E種別.演奏開始,        // 規定値は「演奏開始」
                                                曲ファイルパス = null,
                                                演奏開始小節番号 = 0,
-                                               ドラム音を発声する = false,    // 規定値は false
+                                               ドラム音を発声する = false,  // 規定値は false
                                        };
 
                                        // オプションを定義する。
@@ -643,40 +756,18 @@ namespace SST
                        }
                }
 
-               private void ログインする( string ユーザ名 )
-               {
-                       StrokeStyleT.ユーザ管理.ユーザを選択する( ユーザ名 );
-                       FDK.Log.Info( $"ユーザが選択されました。[{StrokeStyleT.ユーザ管理.現在選択されているユーザ.名前}]" );
-               }
-               private void デバイス情報を出力する()
-               {
-                       using( var stream = new NamedPipeClientStream( "SSTFEditor Viewer Device Information" ) )
-                       {
-                               try
-                               {
-                                       stream.Connect( 1000 );
-
-                                       using( var writer = new StreamWriter( stream ) )
-                                       {
-                                               writer.WriteLine( $"SoundDevice.Delay={StrokeStyleT.Wasapiデバイス.遅延ms.ToString()}" );
-                                               FDK.Log.Info( "デバイス情報を出力しました。" );
-                                       }
-                               }
-                               catch( TimeoutException )
-                               {
-                                       FDK.Log.WARNING( "SSTFEditor ビュアー用パイプへの接続がタイムアウトしました。SSTFEditor が起動していない可能性があります。" );
-                               }
-                       }
-               }
-
-               #region " Win32 API "
-               //-----------------
-               [ System.Runtime.InteropServices.DllImport( "winmm.dll", EntryPoint = "timeBeginPeriod" )]
-               private static extern uint timeBeginPeriod( uint uMilliseconds );
-
-               [System.Runtime.InteropServices.DllImport( "winmm.dll", EntryPoint = "timeEndPeriod" )]
-               private static extern uint timeEndPeriod( uint uMilliseconds );
-               //-----------------
+               #region " バックストア。"
+               //----------------
+               private static SST.フォルダ bs_フォルダ = null;
+               private static FDK.入力.Keyboard bs_キーボード入力 = null;
+               private static FDK.入力.MidiIn bs_MIDI入力 = null;
+               private static FDK.メディア.サウンド.WASAPI排他.ExclusiveDevice bs_Wasapiデバイス = null;
+               private static readonly Random bs_乱数 = new Random( DateTime.Now.Millisecond );
+               private static SST.ユーザ.ユーザ管理 bs_ユーザ管理 = null;
+               private static SST.曲.曲ツリー管理 bs_曲ツリー管理 = null;
+               private static SST.設定.Config bs_Config = null;
+               private static readonly ConcurrentQueue<SST.ステージ.演奏.ビュアーメッセージ> bs_ビュアーメッセージキュー = new ConcurrentQueue<ステージ.演奏.ビュアーメッセージ>();
+               //----------------
                #endregion
        }
 }
index c99b6cb..9cf06d9 100644 (file)
@@ -42,7 +42,6 @@
       <HintPath>..\packages\CSCore.1.1.0\lib\net35-client\CSCore.dll</HintPath>
       <Private>True</Private>
     </Reference>
-    <Reference Include="Microsoft.VisualBasic" />
     <Reference Include="Mono.Options, Version=4.0.0.0, Culture=neutral, processorArchitecture=MSIL">
       <HintPath>..\packages\Mono.Options.4.4.0.0\lib\net4-client\Mono.Options.dll</HintPath>
       <Private>True</Private>
@@ -96,6 +95,7 @@
     <Reference Include="System.Xml" />
   </ItemGroup>
   <ItemGroup>
+    <Compile Include="Win32.cs" />
     <Compile Include="ステージ\演奏\ビュアーメッセージ.cs" />
     <Compile Include="曲\タイトルテクスチャ.cs" />
     <Compile Include="ステージ\選曲\曲パネルビュー.cs" />
diff --git a/StrokeStyleT/Win32.cs b/StrokeStyleT/Win32.cs
new file mode 100644 (file)
index 0000000..cb71b95
--- /dev/null
@@ -0,0 +1,17 @@
+using System;
+using System.Collections.Generic;
+using System.Diagnostics;
+using System.Linq;
+
+namespace SST
+{
+       static class Win32
+       {
+               [System.Runtime.InteropServices.DllImport( "winmm.dll", EntryPoint = "timeBeginPeriod" )]
+               public static extern uint timeBeginPeriod( uint uMilliseconds );
+
+               [System.Runtime.InteropServices.DllImport( "winmm.dll", EntryPoint = "timeEndPeriod" )]
+               public static extern uint timeEndPeriod( uint uMilliseconds );
+
+       }
+}