これいつも忘れるのでメモ。

IsHitTestVisible="False"

CATiledLayerを使ったUIViewの破棄と生成を何度か繰り返していくと、
まれに黒いブロックがフェードインしてくる現象が出ました。

removeFromSuperviewやreleaseなどをして破棄したつもりでも、タイル部分のアニメーションが残り、場合によって黒い(何も描画されていない)タイルが出てくるのだろうと思われます。

しかしこれがまた、どうやって治すかが調べてもさっぱりでして、いろいろ試したところ、
removeFromSuperviewする前ぐらいで
_tiledView.layer.contents = nil;
などとしておくと、黒いブロックは出なくなりました。

ちなみに、UIViewのdeallocでnilを突っ込むと、場合によってタイル更新処理が止まってしまう事もありましたので、タイミングに要注意です。

タイルの更新処理が、ごくまれに止まってしまう現象を確認しました(↑の処置が原因かは確定していませんが)。
黒タイルは放置の方向で・・・。

カスタムコントロールを作ってデザイナへ貼り付けても、InitializeComponent()内でnewされず、実行するとエラーになるケースが頻発しました。

これどうやら、カスタムコントロールのコンストラクタがpublicでないと起こるみたいです。
http://stackoverflow.com/questions/5684784/custom-user-control-not-initialized-in-auto-generated-code

interfaceの実装もpublic固定ですし、デザイナで表示させるプロパティもpublicである必要がありますし(うろ覚え)、データバインディングを行わせるためのプロパティもpublicでないといけません。

外部に公開する気は無いのでinternalをよく使っているのですが、結構こういう制限があって統一しきれないのが残念です。

MFCどっぷり時代からC#のWinFormsをいじりだして半年ぐらい経った気がしますが、ラジオボタンの上手な使い方がどこを調べてもいまいち解りませんでした。

MFCならラジオボタンのどこにチェックが入っているかインデックス値を取得・設定出来るのですが、WinFormsはその辺のサポート機能が一切無いんですよね。
コンボボックスならデータバインディング等も使えていろんなサイトでも紹介されているのですが、ラジオボタンの記事はほとんど見かけた事がありません。

いろいろ試した結果、なんとか着地できたようなので、晒してみたいと思います。

まず、MFCと違い、WinFomrsでは「ラジオボタンのグループ化=どっかのコンテナにまとめて乗せる」というのが大前提です。
(これ最初は違和感有ったのですが。)

逆に言えば、コンテナに対してラジオボタンの便利関数を実装出来れば良さそうです。
C#は拡張メソッドが使えますから、以下のような関数を実装してみました。

/// <summary>
/// ラジオボタンを抽出
/// </summary>
/// <param name="parent"></param>
/// <returns></returns>
internal static IEnumerable<RadioButton> GetRadioButtons(this Control parent)
{
    return parent.Controls.Cast<Control>().Select(x => x as RadioButton).Where(x => x != null);
}

/// <summary>
/// Tagを元にラジオボタンにチェックを入れる
/// </summary>
/// <param name="parent"></param>
/// <param name="tag"></param>
internal static void CheckRadioButtonByTag(this Control parent, object tag)
{
    foreach (var radio in parent.GetRadioButtons())
    {
        if (radio.Tag.Equals(tag))
        {
            radio.Checked = true;
            break;
        }
    }
}

/// <summary>
/// チェックが入ったラジオボタンのタグを取得する
/// </summary>
/// <param name="parent"></param>
/// <returns></returns>
internal static object GetCheckedRadioButtonTag(this Control parent)
{
    foreach (var radio in parent.GetRadioButtons())
    {
        if (radio.Checked)
        {
            return radio.Tag;
        }
    }
    return null;
}

/// <summary>
/// ラジオボタンのチェックイベントを設定
/// </summary>
/// <param name="parent"></param>
/// <param name="action"></param>
internal static void SetRadioButtonCheckedEvent(this Control parent, Action<RadioButton> action)
{
    foreach (var radio in parent.GetRadioButtons())
    {
        radio.CheckedChanged += (sender, _) =>
        {
            RadioButton r = (RadioButton)sender;
            if (r.Checked)
            {
                action(r);
            }
        };
    }
}

あとはFormのLoadイベント辺りで

sizeB5RadioButton.Tag = PaperKind.B5;
sizeA4RadioButton.Tag = PaperKind.A4;
sizeB4RadioButton.Tag = PaperKind.B4;
sizeA3RadioButton.Tag = PaperKind.A3;
sizeGroupBox.CheckRadioButtonByTag([デフォルトの用紙サイズ]);
sizeGroupBox.SetRadioButtonCheckedEvent(radio => [ラジオボタンにチェックが入った時の処理]);

などと呼べば、良いかと思います。
この例は、フォーム上のグループボックスに用紙サイズのラジオボタンがいくつか並んでいるものです。

ラジオボタンのTagにコードであらかじめEnum値などを入れておき、それをやりくりするのがミソです(Enumで無くとも数値でも文字列でも何でも良いのですが)。
イベントに関してもデザイナで一つ一つに割り当てるのは面倒くさいので、コードで一気にやってしまいます。

後からチェックの入った情報を取り出したい時は

(PaperKind)sizeGroupBox.GetCheckedRadioButtonTag()

などとすれば良いです。

タイプセーフ的にあんまりなコードですが、いちいちフォームでラジオボタンのチェックプロパティを見に行ってチェックされていたら云々、と書くよりは良いと思うのですがどうでしょうかね。

ラジオボタンの定石をご存知の片がいらっしゃいましたら、こっそり教えてもらえると助かりますm(_ _)m

Activity(またはView)の状態保存・復元はonSaveInstanceStateとonRestoreInstanceStateとで行われますが、これのテストが困難だったりします。

(ちょっと憶測も入っていますが)Activityを呼び出したり戻したりすればonSaveInstanceStateとonRestoreInstanceStateが呼ばれるので、コードは一応通るのですが、大抵の場合はストレージ(?)への書き出し・読み込み処理は走らず、メモリ内のキャッシュデータを読み書きしているだけです。

ですが、いざという時(メモリ不足時?)には実際にストレージへの書き出し・読み込み処理が走り、いままで何の問題も無かった箇所で落ちる、何てことが起こりえます。

特に、カスタムViewを作り、Parcelオブジェクトを使って読み書きしている部分は、普段は処理が走らないので厄介です。
参考:
Viewの状態を保存する
http://ym02.blogspot.com/2010/08/view.html

iPhoneだとシミュレータでメモリ不足時のシミュレートが出来るようになっていますが、Androidはそのような事が出来るのでしょうか?
(知っている人いたら教えて下さい)

そこで、擬似的にこの辺りのテストを行うコードを書いてみました。

private void test()
{
	Parcel p1 = Parcel.obtain();
	Bundle b1 = new Bundle();
	onSaveInstanceState(b1);
	p1.writeBundle(b1);
	
	byte[] bytes = p1.marshall();
	
	Parcel p2 = Parcel.obtain();
	p2.unmarshall(bytes, 0, bytes.length);
	p2.setDataPosition(0);
	Bundle b2 = p2.readBundle();
	b2.setClassLoader(getClassLoader());
	onRestoreInstanceState(b2);
}

ただ、このコードによる保存・復元処理が正常時と同じかどうかは保障できません(^^;
また、私自身javaの経験が浅く、ClassLoader辺りはよく理解していません・・・。
(でもsetClassLoaderしないと、うまくparcelが取り出せないんですよね)

1 2 3 4 5 6 10

記事検索

アーカイブ