年別アーカイブ: 2009年

42件の投稿

Spreadの工夫 2

前回の「clsShtData」の主要な利用価値は、[コード]で[デザイン]を決めることと、次の二つの機能です。

一つ目は、列名に対して列番号(Index)を返す仕組みです。

    SpdIdx賃料 = .画面設定(multiHeader, 非表示, True)

では、SpdIdx賃料は実は次のように定義されています。

    Private SpdIdx賃料 As Dictionary(Of String, Integer)

例えば、 SpdIdx賃料(“金額”)の値は2、SpdIdx賃料(“税区分”)の値は3を返します。
同じ列名があるときは、2番目以降は[金額_2]等のように出現番号を付けて返します。
従って、SpdIdx賃料(“金額_2”)の値は8、SpdIdx賃料(“税区分_2”)の値は9を返します。
Spreadでは列や行の指定にはIndex(整数)を使いますが、これをハードコードにすることの煩わしさはプログラムを書く人なら分かると思います。

重宝しているもう一つは、Spreadの幅を計算するものです。まず、[行見出し]の幅で、シートの[RowHeader.Columns](一般に複数列)の幅です。次はシートの表示列(非表示ではなく)の幅の全合計で、前回のコードでは次式で設定しています。

    New clsSpdHeader("コード", 55, SpdIntCellType, False)

さらにバーティカルBarがあればその幅で、Spread の[VerticalScrollBarWidth]プロパティで得られます。
最後はSpreadの両サイドの[BorderStyle]です。
これらをすべて足し合わせてものが必要なSpread幅です。
これによってピタッとしたSpreadを作成できます。

このクラスを継承したクラスもいくつか作っています。
例えば、消費税や管理料の計算をするクラスもその一つです。

Spreadの工夫

私は米FarPoint社のSpreadを多用しています(Excelに似たプログラム開発用シート)。

予断ですが…
このSpreadは日本ではGrapeCity社が日本語版を販売しています。
原版に比べて日本語版は倍近く高価です。
原版はマニュアルやデザイン画面のメニューが英語であることを我慢すれば、使用には何も問題はありません。
ところが、最近このFarPoint社はGrapeCity社に買収されたようです(多分)。

このSpreadでは沢山の工夫をしています。

まず、
Spreadの列幅やCellTypeの設定等々様々な設定をデザイン画面で設定すると、結構手間がかかります。
しかも開発の途中でシートのデザインの変更が多々ありますので、その都度デザイン画面でやり直しをやっていては大変非効率です。
そこで私は、クラス[clsShtData]を作ってこれらの処理を全部このクラスに任せています。
パフォーマンスの検証はしていません。
少なくとも最終版の手前まではこれらの設定をコードで書き、最終的にパフォーマンスを考慮した方法に置き換えるのも一つだと思います(私の経験では特にパフォーマンスが悪いとは思いません)。

Spread2
        Sht賃料 = New clsShtData(FpSpd賃料, 0, True, True, 賃料表示行数)
        With Sht賃料
            With .P_Heading
                .Add(New clsSpdHeader("コード", 55, SpdIntCellType, False))
                .Add(New clsSpdHeader("科目名", 100, SpdStrCellType, False))
                .Add(New clsSpdHeader("金額", 80, SpdComCellType, True))
                .Add(New clsSpdHeader("税区分", 55, cmbcell税区分, False))
                .Add(New clsSpdHeader("消費税", 60, SpdComCellType, True))
                .Add(New clsSpdHeader("合計", 80, SpdComCellType, True))
                .Add(New clsSpdHeader("送金", 40, Spdchkcell, False))
                .Add(New clsSpdHeader("料率", 40, SpdDblCellType, False))
                .Add(New clsSpdHeader("金額", 60, SpdComCellType, True))
                .Add(New clsSpdHeader("税区分", 55, cmbcell税区分, False))
                .Add(New clsSpdHeader("消費税", 60, SpdComCellType, True))
                .Add(New clsSpdHeader("合計", 60, SpdComCellType, True))
                .Add(New clsSpdHeader("賃料明細ID", 60, SpdIntCellType, False))
            End With

            ' VB2008仕様
              Dim multiHeader(,) = New Object(,) {{"金額", 1, 4, "金額"}, {"送金", 1, 6, "管理料"}}
            Dim 非表示() = {"賃料明細ID"}
            SpdIdx賃料 = .画面設定(multiHeader, 非表示, True)
        End With

このコードで、Spread[FpSpd賃料]のインデックス[0]のシートで、行数は[賃料表示行数]に設定、合計行を作成し、指定する列で合計を計算表示します。

[P_Heading]では次の内容を設定しています。
列名は、[コード]、[科目名]、[金額]等で、列幅はそれぞれ[55]、[100]、[80]で、データ型はそれぞれ[SpdIntCellType]、[SpdStrCellType]、[SpdComCellType]であり、[金額]列は合計を計算します。

[画面設定]メソッドでは、[P_Heading]の内容で各シートのスタイルを設定しますが、このとき列名を表示するヘッドの部分は2行にし、列名[金額]から4列分に分類名[金額]、同じく[送金]から6列に分類名[管理料]を付けています。
また、[賃料明細ID]の列は非表示にしています。

[clsShtData]ではそのほか様々な処理をしています。
一般的に参考になることを次回紹介します。

複数コントロールのセットを何度も使う

今、郵便番号を複数の画面で検索する場合を想定します。
郵便番号を入力して[住所検索]ボタンをクリックすると住所を表示し、住所を入力して[郵便番号検索]ボタンをクリックすると郵便番号を表示するというものです。

YubinBango

ここには、二つのTextBoxと二つのボタン、すなわち郵便番号を入出力するTextBox(あるいはMaskedTextBox)、住所を入出力するTextBox 、入力した郵便番号から住所を検索出力するためのボタン、逆に住所から郵便番号を検索出力するためのボタンがあります。

いまここで、ボタンにはそれぞれ郵便番号から住所を検索出力する、逆に住所から郵便番号を検索出力するプログラムが対応しているものとします。

あちこちの郵便番号検索画面で、これらのコントロールを配置しボタンのイベントプロシージャをその都度書くのは嫌です。

一つの解決策は、上のすべてのコントロールを一つにカスタムコントロールにする手があります。
が、少し大げさなので、次のようなクラスを一つ作っています。
主眼は、このクラスのインスタンスを作るとき、クライアントのコントロールをByRefで渡すこと、渡されたコントロールのイベントをこのクラスの中に定義することです。

Public Class cls住所検索
    Friend WithEvents m_btn住所検索 As Button
    Friend WithEvents m_btn郵便番号検索 As Button

    Private m_msk郵便番号 As MaskedTextBox
    Private m_txt住所 As TextBox

    Sub New(ByRef btn住所検索 As Button, ByRef btn郵便番号検索 As Button, _
        ByRef msk郵便番号 As MaskedTextBox, ByRef txt住所 As TextBox)

        m_btn住所検索 = btn住所検索
        m_btn郵便番号検索 = btn郵便番号検索
        m_msk郵便番号 = msk郵便番号
        m_txt住所 = txt住所
    End Sub

    Private Sub btn住所検索_Click(ByVal sender As Object, ByVal e As System.EventArgs) _
        Handles m_btn住所検索.Click

        Using frm As New frm住所取得
             frm.Called郵便番号2住所(m_msk郵便番号.Text)
            If frm.DialogResult = Windows.Forms.DialogResult.OK Then
                m_txt住所.Text = frm.P_住所
             Else
                MsgBox("この郵便番号は、有効ではありません。郵便番号を変更してください。", _
                    MsgBoxStyle.Exclamation, "住所検索")
            End If
        End Using
    End Sub

    Private Sub btn郵便番号検索_Click(ByVal sender As Object, ByVal e As System.EventArgs) _
        Handles m_btn郵便番号検索.Click
        Using frm As New frm郵便番号取得
            frm.Called住所2郵便番号(m_txt住所.Text)
            If frm.DialogResult = Windows.Forms.DialogResult.OK Then
                m_msk郵便番号.Text = frm.P_郵便番号
            Else
                MsgBox("この住所は、有効ではありません。住所を変更してください。", _
                    MsgBoxStyle.Exclamation, "郵便番号検索")
            End If
        End Using
    End Sub
End Class

これを使う画面のコードは、次のようになります。

    Private inst住所検索 As cls住所検索

   Private Sub frm_Load(ByVal sender As Object, ByVal e As System.EventArgs) Handles Me.Load
        inst住所検索 = New cls住所検索(btn契約者住所検索, btn契約者郵便番号検索, _
                msk契約者郵便番号, txt契約者住所1)
    End Sub

すなわち、各画面ではこのクラスのインスタンスを作るだけで、処理はすべてこのクラスが行います。
私はこのような仕組みをよく使います。
たとえば、氏名を漢字で入力するとそのふりがなを別のTextBoxに表示する場合も同様です。

胆嚢結石 6

手術を受けると色々なことを知ります。

麻酔下で長時間手術すると排尿ができなくて、腎臓等に異常が起こるためと思われますが、
尿道から尿カテーテルと称する管を膀胱(多分?)まで突っ込まれ、膀胱に溜まった尿は自動的に体外に排出されます。
本人は、排尿の心配はないのですが、その分身動きがとれません。

私には気管支拡張症があり、普段でも痰が多いのですが、手術をするとこれが増えます(これも麻酔のせいらしいです)。
何度も咳き込み、そのため傷が痛いのか、咳のため腹筋が痛いのか分かりません。今回の手術で一番悩まされたことです。

さて、12日4時前手術が終わってから一晩ベットに身動きできない状態でいました。

うとうとしていた夜中、同室の十二指腸を切除した患者が(すい臓に腫瘍ができ、そのために十二指腸も切除したようです)大騒ぎを始めました。
彼は確か5日辺りに手術したのですが、ゲップが出そうで出なかったり、腹がはったり、その上シャックリが止まらず大分苦しいらしいのです。
私は尿カテーテルを付けていて身動きができません。
部屋の入り口付近で「ガタンガタン、ゴロンゴロン」歩き回っています。
看護師がきて色々手当てをしていましたが、夜中の2時頃また騒いでいます。
挙句の果てに、同室の別の患者のカーテンに入り込んで、その患者が看護師を呼び、疲れ果てたのかその後眠ったようです。

後で分かったのですが、その夜は早い段階で睡眠薬を飲ませた(点滴かもしれません)ようで、その副作用で異常行動をしたようです。
本人はその夜の行動を覚えていないようです。

私と同病で胆嚢を摘出した人は、その後も「痛い」といって、痛み止めや睡眠薬を要求していましたが、
「いったいどうなっているのやら」と訳が分かりません。

患者が退院して1時間もすると次の患者が運び込まれてきます。
この2週間で私を含めて8人の患者がこの部屋で入院生活をすごします。

退院の会計を待っている間、寒々しい曇天下、病室から眼下の小川を見ていたら美しいカワセミが飛んできました。
少しハッピーな気分になりました。

色々ありましたが、予定より2日遅れて昨17日無事我が家に戻ってきました。
まだ抜糸がすんでいませんが、酒も運動もOKが出ましたので、体力の回復に努めたいと思います。

胆嚢結石 5

無事生還–というほど大げさではないか。

11日主治医から手術の説明を受けました。
病名:胆嚢結石、手術名:腹腔鏡下胆嚢摘出術。

「お腹に数箇所穴を開けそこから胆嚢を摘出しますが、場合によっては開腹手術になります」
「同意であれば署名してください」

こんな場合不同意などないでしょう。

手術は12日正午過ぎに決まりました。

12日は朝から点滴を受けました。手術ではこれに麻酔薬や抗生物質を流します。
T字帯と称するフンドシの上は、手術着一枚になります。

先日も書きましたが、同室に同じ病気の患者がいて、盛んに痛みを訴えていて重篤そうです。
9時半過ぎ彼が手術に行った気配です。
が、1時間足らずで帰ってきました。

本当に簡単な手術なのだ。
私は別の手術が入ったのか、「2時以降です」と手術の変更を告げられました。

まあ私は大したことはないのだし、「どうぞどうぞ」という気分。

ところが1時少し前、「1時から手術です」と突然看護師がやってきました。
心の準備もないままに、主治医に先導されて手術室に入りました。

手術室専門の看護師から、「気分はいかがですか」と聞かれ「気分言いわけないでしょう」と本音。
手術台に横になると、直ちに左腕に血圧計等の検査器具がつけられたようです。

右腕は点滴がつけられています。
麻酔医が「麻酔を入れます」という言葉を最後に手術はまったく記憶がありません。

耳元で麻酔医が私の名前を呼ぶ声で目覚めましたが、喉に痰が溜まっていて声がでません。
たぶん私は必死に喉を指さしたと思います。

すぐさま痰を吸いだしてくれたので、やっと声をだすことができました。

ストレッチャーに乗せられて手術室から出ようとしたとき、主治医が椅子に座り込んで、「大変だった」とつぶやいていました。
「今何時ですか」と私。「3時40分です」と誰か。

「え。大手術だったのだ」(本当の大手術ではないのですが、私としては…)
後で分かったのだが、私の胆嚢は他の臓器に癒着して、その臓器から胆嚢を切り離すのが大変だったのだという。
「何度も開腹しようと迷いました」と主治医。
「がんばってもらってよかった」とつくづく安堵したところです。