コントロールの均等配置(フレーム周期の描画)
Silverlight のサンプル
(ブラウザのウインドウサイズを変えて見てください)
Silverlight のサンプル2
マウスホイールでのズーミング、パネルのドラッグを追加しました。
「コントロールの動的配置」は、ContentPresenterクラスとインターバルタイマで自律的に移動するコントロールを作って見ました。今回は、フレーム周期のイベントで表示を更新する方法を試します。目的はコントロールを移動させながら均等に配置することです。配置しているのは単なる図ではなく、コントロールです。クリックのイベントを処理したり、いろいろなプロパティを設定することができます。
※ サンプル画像のフルーツ画像は、「WEB素材-Memory」の画像を使わせていただきました。
http://ao777.blog64.fc2.com/blog-category-24.html
から part1fruit.zip 中の png を列挙しています。
動きながら ==> 均等に並ぶ |
|
|
1.目標と、何ができたか
1.1.均等配置
概ね思うようになったのですが、目標はもっと強力なものです。
上の図では、四隅が有効に使われていません。個々の図(マーク)の間隔が狭い場合は四隅も使って矩形に配置し、間隔が広い場合は円形に並ぶようになるのが理想的だと考えます。また、ウインドウを広げた場合、適切なマークの間隔以上には広がらないことも必要です。
1.2.ランダムな順序
出来たものは、毎回マークの表示順が異なります。視覚的効果としては良いのですが、実用的なアプリケーションのボタンとして使うには順番が変わらないものも必要だと思います。
必ずしも大きくマークが移動する必要はないので、納まるべき位置に向かって揺れて移動するような動作の方が実用的かもしれません。
1.3.ライブラリ化
ネットワーク構造のデータを図示するようなライブラリを作りたいのですが、やっと動作の仕組みが分かってきたところで道は遠いようです。
2.動作の仕組み
2.1.PanelクラスとCompositionTarget.Rendering
Panelを継承したクラスで、CompositionTarget.Renderingにハンドラを設定するとフレーム周期でハンドラが呼び出されるようになると言う仕組みを使います。
フレーム周期ですが大まかに実測すると16msでした。モニタのリフレッシュレート由来と考えています。モニタのリフレッシュレートを 60 - 80Hz とすれば処理間隔は 12-17ms で、差を無視します。厳密な処理や、高性能モニタがあるならリフレッシュレートを読み取って処理する必要があるでしょう。
負荷が高いときにはどうなるのかも気にはなりますが、見た目の問題だけなので追及しないことにします。
2.2.スタティックなイベントハンドラ登録
インテリセンスの助けを借りて、
CompositionTarget.Rendering +=
まで入力して、Tabキーを2回押すとハンドラのメソッドまで作ってくれます。
このメソッドで、マークの位置を移動します。
ここで気が付いたのは、CompositionTargetはインスタンシングしたオブジェクトではないので、Rendering はスタティックだと言うことです。名前空間で1義の登録場所があると言うことです。
このスタティックなイベントハンドラの登録場所を使う方法は、マークがクリックされたことをアプリケーションに通知するイベントハンドラの登録場所の確保にも利用できます。
個々のマークにハンドラを設定しなくても良くなります。
2.3.マークの作り方
「コントロールの動的配置」と同様、ContentPresenterクラスを継承して作ります。
3.アプリケーション
Panel1 に描画処理を作ることにして、XAML の Grid に Panel1を割り当てます。
また、マークがクリックされたとき、どのマークか分かるようにステータスバーに識別子を表示します。
- public partial class Window1 : Window
- {
- public Window1()
- {
- InitializeComponent();
- }
- private void grid1_Loaded(object sender, RoutedEventArgs e)
- {
- Panel1 panel = new Panel1();
- panel.Background = Brushes.Bisque;
- grid1.Children.Add(panel);
- MarkBase.MouseClickHandler += new MarkBase.MouseEvent(MarkBase_MouseClickHandler);
- }
- void MarkBase_MouseClickHandler(object sender, RoutedEventArgs e)
- {
- statusBar1.Text = ((MarkBase)sender).Identifier;
- }
- }
Panel1クラスは、Panelを継承します。ライブラリ化したい部分と未分化で、計算部分を除いてあります。要点は、コンストラクタでマーク図形のオブジェクトを作成、追加して、CompositionTarget.Renderingにハンドラを登録することです。
ハンドラは、フレーム周期で呼び出され、各マークの表示位置を移動します。
移動しなくなり、収束したらハンドラの登録を削除します。
マークは、ContentPresenterクラスから派生したMarkBaseクラスを継承しているので、Childrenからマークを認識できます。
- class Panel1 : Panel
- {
- public Panel1()
- {
- //英字の入った丸いマークを描画
- string labels = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
- for (int i = 0; i < labels.Length; i++)
- {
- MarkBorder mb = new MarkBorder(labels.Substring(i, 1));
- this.Children.Add(mb);
- }
- //フルーツのマークを描画
- string[] fns = Directory.GetFiles(©"..\..\png", "*.png");
- foreach (string fn in fns)
- {
- MarkImage mi = new MarkImage(fn);
- this.Children.Add(mi);
- }
- //リサイズいベント処理の追加
- this.SizeChanged += new SizeChangedEventHandler(Panel1_SizeChanged);
- //フレーム周期のイベント処理を追加
- if (compositionTargetRendering <= 0)
- {
- compositionTargetRendering++;
- CompositionTarget.Rendering += new EventHandler(CompositionTarget_Rendering);
- }
- }
- //----------------------------------------------------------
- //CompositionTarget.Renderingへのハンドラ設定を2重にしないように
- int compositionTargetRendering = 0;
- //----------------------------------------------------------
- //リサイズ時に計算したウインドウ表示サイズに依存した値
- Point panelCenter; //パネルの中心の座標
- double maxVectorLength; //パネルの中心からの距離の最大値
- //----------------------------------------------------------
- #region マークの位置を計算するメソッド
- #endregion
- //----------------------------------------------------------
- //ウインドウのリサイズのイベント
- void Panel1_SizeChanged(object sender, SizeChangedEventArgs e)
- {
- ChangePanelCenter();
- }
- //レンダリング・イベント
- void CompositionTarget_Rendering(object sender, EventArgs e)
- {
- //MarkBase型を持つものだけ抽出。追加削除を考慮して毎回算出。
- int mbCount = 0;
- int[] index = new int[this.Children.Count];
- for (int i = 0; i < this.Children.Count; i++)
- if (this.Children[i] is MarkBase)
- index[mbCount++] = i;
- Vector[,] F = new Vector[mbCount, mbCount];
- for (int i = 0; i < mbCount; i++)
- {
- MarkBase i_mb = (MarkBase)this.Children[index[i]];
- F[i, i] = new Vector();
- for (int j = (i + 1); j < mbCount; j++)
- {
- MarkBase j_mb = (MarkBase)this.Children[index[j]];
- F[i, j] = GetRepulsiveForce(RandomVecorIfZero(i_mb.Location - j_mb.Location), mbCount);
- F[j, i] = -F[i, j];
- }
- }
- bool changed = false;
- for (int i = 0; i < mbCount; i++)
- {
- MarkBase mb = (MarkBase)this.Children[index[i]];
- Vector v = new Vector();
- for (int j = 0; j < mbCount; j++)
- if (j != i) v += F[i, j];
- v += GetSpringForce(RandomVecorIfZero(
- mb.Location - panelCenter), mbCount);
- v += MaxMovement(mb.Location);
- changed |= updateMark(mb, v);
- }
- if (!changed)
- {
- //変化が無くなったらレンダリングイベントを停止
- CompositionTarget.Rendering -= new EventHandler(CompositionTarget_Rendering);
- compositionTargetRendering--;
- }
- }
- }
丸に英字のマークは、Borderクラスでできています。
- class MarkBorder : MarkBase
- {
- public MarkBorder(string label)
- : base(label)
- {
- this.Cursor = Cursors.Hand;
- ContentPresenter CP = new ContentPresenter() { Content = label };
- base.Content = new Border()
- {
- Background = Brushes.Aquamarine,
- Padding = new Thickness(5),
- CornerRadius = new CornerRadius(16),
- BorderThickness = new Thickness(2),
- BorderBrush = new SolidColorBrush(Colors.Gray),
- Child = CP
- };
- }
- }
果物のマークは、Imageクラスでできています。
- public MarkImage(string src)
- : base(src)
- {
- this.Cursor = Cursors.Hand;
- Image img=new Image();
- BitmapImage bm = new BitmapImage();
- bm.BeginInit();
- bm.UriSource = new Uri(src, UriKind.Relative);
- bm.CacheOption = BitmapCacheOption.OnLoad;
- bm.EndInit();
- img.Width = 32;
- img.Height = 32;
- img.Source = (BitmapSource)bm;
- base.Content = img;
- }
|