- 締切済み
サブフォーム帳票形式上での2つのコンボボックスの連携
サブフォーム帳票形式上での2つのコンボボックスの連携 テーブルとして以下4つ tbl01_受注 受注ID(主キー) 受注日 名前 電話番号 tbl02_明細 明細ID 受注ID(tbl01_受注IDとリレーション) カテゴリ 商品名 tbl03_カテゴリ カテゴリID カテゴリ tbl04_商品名 商品ID カテゴリID 商品名 フォームとして以下メインとサブで構成 f01_受注フォーム(メインフォーム) レコードソース→tbl01_受注 f02_明細フォーム(サブフォーム) レコードソース→tbl02_明細 添付画像のように 1件ごとの受注につき受注フォームが増えていきます 1件の受注にたいしてサブフォームに商品を入力して行きます、 商品名が多数ありますので コンボ1でtbl03のカテゴリを選び tbl04からそのカテゴリIDとマッチした商品名を抽出して コンボ2に表示したいと考えています。 色々参考にしましたがどれもうまく行かない状況です。 たとえばコンボボックスに反映されず空白になったり コンボボックスの内容を変更すると全ての行の値が 同じになった なかなかうまく行きません。 サブフォームでコンボボックス1でカテゴリ選択すると コンボボックス2にカテゴリで絞り込んだ内容が選択できるように したいのです。 そして次の行も同じように商品を入力したいのです。 どなたか助けて下さい、お願いします。
- みんなの回答 (6)
- 専門家の回答
みんなの回答
- piroin654
- ベストアンサー率75% (692/917)
>何かご指摘して頂けると幸いです。 ファイルを見ます。しばらく。
- piroin654
- ベストアンサー率75% (692/917)
【続き】 (8) 「f01_受注フォーム」の調整 「f01_受注フォーム」をデザインビューで開き、 フォームの上部の適当なところにコマンドボタン を設定します。 コマンドボタンの名前:cmd商品 コマンドボタンの表示:商品 フォームに以下を貼り付け保存します。 Private Sub cmd商品_Click() DoCmd.OpenForm ("f_商品") End Sub (9) 「f02_明細フォーム」の調整 「f02_明細フォーム」のコード表の全てを 以下に置き換えて、保存してください。 Option Compare Database Option Explicit Dim lnNum As Long Private Sub カテゴリ_AfterUpdate() Dim rsm As Recordset Set rsm = Me.RecordsetClone '明細IDの採番 If IsNull(Me!明細ID) Then If rsm.RecordCount > 0 Then rsm.MoveFirst lnNum = rsm!明細ID + 1 Do Until rsm.EOF If rsm!明細ID >= lnNum Then lnNum = rsm!明細ID + 1 End If rsm.MoveNext Loop Me!明細ID = lnNum Else '明細IDは1から始める Me!明細ID = 1 End If 'カテゴリだけを変更したままにすることの回避 Else '変更しない場合 If MsgBox("変更しますか", vbYesNo) = vbNo Then Me.Undo '変更する場合 Else Me!商品ID = 0 Me!商品名 = Null End If End If rsm.Close: Set rsm = Nothing End Sub Private Sub 商品ID_AfterUpdate() Me![tbl02_明細.商品名] = Me![tbl04_商品名.商品名] End Sub Private Sub 商品ID_Enter() Me!商品ID.Requery End Sub Private Sub 商品名_AfterUpdate() DoCmd.RunCommand acCmdSaveRecord End Sub (10) テーブル「tbl02_明細」の調整 「tbl02_明細」の「明細ID」を数値型に変更し、 既定値の「0」を削除してください。また、 受注ID、「カテゴリID」、「商品ID」の既定値 の「0」も削除してください。 以上で設定は終わりです。 なお、「f02_明細フォーム」の「明細ID」の並びを 設定していないので、「f02_明細フォーム」の レコードソースを開き、「明細ID」の「並び」を 「昇順」に設定してください。 一応、明細IDはオートナンバーになっていましたが、 明細IDの必要性は各受注伝票内での並びならばとくに オートナンバーである必要は無く、むしろ将来の 拡張性を考えればオートナンバーでないほうがいいと 思いこのようにしました。どうしてもオートナンバー でなければならない理由があるのならオートナンバー 戻します。その場合は(9)のコードを少し変更します。 少し大変かもしれませんがじっくり作ってみてください。 同じような質問は過去にもあったのだと思いますが たぶんいろいろな設定を伝えるのが大変で回答は ほとんどついていなかったのでは思います。 以上の内容でも不十分なところはあるのですが、 基本的なところは押さえてあるつもりです。 この受注フォームは少し改造すると、納品フォームに 変更できます。 また、本来は現在使用しているテーブルは作業テーブル にして、納品確認が終了したら受注台帳と受注明細台帳 に転記して、空にしておくのが一般的です。 なお、納品確認は「tbl01_受注」にYes/No型の納品確認 のフィールドを作り、メインフォームにこれをレコードソース とするチェックボックスを作り、納品が終了したら チェックをいれておきます。 台帳への転記もこのチェックを目安にして受注、および受注明細 を転記することが可能です。その場合、各台帳はオートナンバー になっているフィールドはすべて数値型にして同じ構造のテーブルに しておきます。転記される台帳側がオートナンバーのままだと エラーをおこすからです。 このようにしてシステムを作っていきます。
- piroin654
- ベストアンサー率75% (692/917)
商品登録時の間違いや商品データの管理 などを検討しましたが、やはりコンボボックスを 連動させる形でのコンボボックスでのカテゴリや 商品名の随時登録は間違いが起こりやすく、また 結構面倒な方法になるので、あっさりと 商品登録フォームを作ったほうがいいのでは と思います。また、一般的にはこのような 方法を取るのが普通です。 一応、f01_受注フォームから商品登録フォームを 呼び出し、登録したカテゴリ、商品は登録後 商品登録フォームを閉じるときにf01_受注フォームの コンボボックスに反映するようにします。 少し大変ですが作ってみてください。 (1) 準備 f01_受注フォームのf02_明細フォームを表示している コントロールの名前をf02_明細フォームに変更してください。 初期値は埋め込み0のような名前になっています。 (2) 商品登録フォームの作成 名前:f_商品 新しいフォームに二つのサブフォーム表示コントロール を設定します。 左のコントロールの名前を「fカテゴリ」、 リンクオブジェクトを「f_カテゴリ」にします。 右のコントロールの名前を「f_商品詳細」にします。 リンクオブジェクトを「f_商品詳細」 リンク子フィールドを「カテゴリID」 リンク親フィールドを「tx隠しカテゴリ」 にします。 各表示コントロールの幅は少し広めにしておいてください。 後で幅は調整してください。 「f_商品」の上部にテキストボックスを1個設定します。 テキストボックスの名前:tx隠しカテゴリ このテキストボックスのコントロールソースに、 =[Forms]![f商品]![fカテゴリ].Form.カテゴリID を設定します。 「可視」を「いいえ」に設定します。 「f_商品」の閉じるときのイベントに以下を 設定します。 Private Sub Form_Close() 'f02_明細フォームの各コンボボックスの再設定 If IsLoaded("f01_受注フォーム") Then Forms!f01_受注フォーム!f02_明細フォーム.Form.カテゴリ.Requery Forms!f01_受注フォーム!f02_明細フォーム.Form.商品ID.Requery End If End Sub 保存して閉じます。 (3) 「f_カテゴリ」の作成 新規のフォームを作り、そのレコードソースに SELECT DISTINCTROW tbl03_カテゴリ.カテゴリID, tbl03_カテゴリ.カテゴリ FROM tbl03_カテゴリ; を貼り付け保存してください。フォームの詳細の ところにテキストボックスを二つ置き、それぞれの コントロールソースを「カテゴリID」、「カテゴリ」 としてください。ラベルは詳細の中には必要ありません。 名前を「f_カテゴリ」として、保存して閉じます。 (4) 「f_商品詳細」の作成 新規のフォームを作り、そのレコードソースに 「tbl04_商品名」を指定します。 (3)と同様にして、フォームの詳細部分に、 「商品ID」、「カテゴリID」、「商品名」を コントロールソースにするテキストボックスを 設定してください。 (5) 「f_商品」の調整 「f_商品」をデザインビューで開き、 サブフォーム表示コントロールにそれぞれ 「f_カテゴリ」、「f_商品詳細」が 表示してあるか確認してください。 (6) 以下のコードを標準モジュールに 貼り付け保存 Function IsLoaded(ByVal strFormName As String) As Boolean ' 指定したフォームがフォーム ビューまたはデータシート ビューで開かれている場合 True を返します。 Const conObjStateClosed = 0 Const conDesignView = 0 If SysCmd(acSysCmdGetObjectState, acForm, strFormName) <> conObjStateClosed Then If Forms(strFormName).CurrentView <> conDesignView Then IsLoaded = True End If End If End Function (7) 誤作動防止のための設定 「f02_明細フォーム」をデザインビューで開き、 名前が「商品名」のテキストボックスの プロパティの「編集ロック」を「はい」に 設定してください。 これは商品テーブルにない商品に名前が 変更されてデータがおかしくなる事を 防止します。すべての商品の設定は 「f_商品」で行ないます。 【字数制限のため続く】
- piroin654
- ベストアンサー率75% (692/917)
>あとコンボボックスに無い内容でも >手入力で記入できるものを望んでいます。 の機能を入れていなかったので確認ですが、 「tbl04_商品名」の「商品ID」と 「tbl03_カテゴリ」の「カテゴリID」はオートナンバーですか。 あるいは違った方法でそれぞれを決めていますか。 その場合、それぞれの型は数値型ですか、 あるいは、テキスト型ですか。
お礼
piroin654様 引き続きご対応有難う御座います、 >「tbl04_商品名」の「商品ID」と >「tbl03_カテゴリ」の「カテゴリID」はオートナンバーですか。 それぞれオートナンバーで数値型で考えています。 上記形式以外でやり易いのでしたら 変更は可能です。 宜しくお願い致します。
- piroin654
- ベストアンサー率75% (692/917)
テーブル「tbl02_明細」の構造を一部変更 します。 フィールド名「カテゴリ」を「カテゴリID」にしてください。 型はたぶん数値型だろうと思いますが。 一応、質問の内容でサンプルを作ってみました。 サブフォームに配置するコントロールは6個に なります。 以下サブフォームの設定です。 (1) まず、以下のクエリを作成します。名前は「Q受注明細」です。 SELECT tbl02_明細.明細ID, tbl02_明細.受注ID, tbl02_明細.カテゴリID, tbl02_明細.商品ID, tbl02_明細.商品名, tbl04_商品名.商品名 FROM tbl04_商品名 INNER JOIN tbl02_明細 ON tbl04_商品名.商品ID = tbl02_明細.商品ID; (2) サブフォームのレコードソースを以下にします。 SELECT DISTINCTROW Q受注明細.明細ID, Q受注明細.受注ID, Q受注明細.カテゴリID, Q受注明細.商品ID, Q受注明細.tbl02_明細.商品名, Q受注明細.tbl04_商品名.商品名 FROM Q受注明細; 上記で選択されているフィールドをもとにサブフォームに テキストボックスを配置します。その中の「カテゴリID」 と「商品ID」を表示するコントロールはコンボボックスにします。 以下、コントロールソースと名前の関係です。 コントロールソース 名前 明細ID 明細ID 受注ID 受注ID カテゴリID カテゴリ 商品ID 商品ID tbl02_明細.商品名 商品名 tbl04_商品名.商品名 dum商品名 (3) 「tbl04_商品名.商品名」を表示するテキストボックス「dum商品名」 の幅を最小に縮めてプロパティで「可視」を「いいえ」にします。 「dum商品名」を「商品名」の位置の真ん中に置き、上から 「商品名」をかぶせます。 (4) 「カテゴリ」の値集合ソースに以下を貼り付けます。 SELECT tbl03_カテゴリ.カテゴリID, tbl03_カテゴリ.カテゴリ FROM tbl03_カテゴリ; 「カテゴリ」のプロパティで 列数:2 列幅:0cm,3cm 連結列:1 リスト幅:3cm としてください。 (5) 「商品ID」の値集合ソースに以下を貼り付けます。 SELECT tbl04_商品名.商品ID, tbl04_商品名.商品名 FROM tbl04_商品名 WHERE (((IIf(IsNull([カテゴリ]),True,[カテゴリID]=[カテゴリ]))=True)); 「商品ID」のプロパティで 列数:2 列幅:3cm,3cm 連結列:1 リスト幅:6cm としてください。 (6) 「商品ID」の更新後処理に以下を設定します。 Private Sub 商品ID_AfterUpdate() Me![商品名] = Me![dum商品名] End Sub (7) 「商品ID」のフォーカス取得時に以下を設定します。 Private Sub 商品ID_Enter() Me!商品ID.Requery End Sub 以上で設定は終了です。肝だらけですが、もっとも 肝になるのは(5)の値集合ソースの設定です。 サブフォーム内でのコンボボックスの連動は なかなか分かりづらいところがありますが、 コツをつかめばどうにでもなります。 何かあれば補足してください。
お礼
piroin654様 書き込み誠に有難う御座います、 ほんと無人島に人が尋ねてきた感じです!! しかも頼もしそうな人が! 頂いたご指示に従いこれから作業を始めます、 ただ私、飲み込みが遅い方ですので piroin654様のご意見を再現するのに 少しお時間をお掛けすると思いますが 必ず結果をご報告いたします。 今しばらくお待ち頂けますようお願い申し上げます
- piroin654
- ベストアンサー率75% (692/917)
なかなかレスがつかないようですが、 (1) サブフォームのレコードソースは どのようになっていますか。 (2) サブフォームはデータシート形式 ですか、あるいは帳票形式ですか。 あるいはどちらでもかまいませんか。
お礼
piroin654様 レスが無くあきらめ掛けていました、 (1)の回答 サブフォームは tbl02_明細を参照しています レコードの内容は以下の4個です 明細ID(オートナンバー) 受注ID(tbl01_受注IDとリレーション) カテゴリ 商品名 単純な受注伝票をイメージしています メインフォームにお客様の名前と電話番号があり サブフォームに買った商品を1行づつカテゴリと商品名を記入していく感じです サブフォームの1行ごとに コンボボックス1としてカテゴを選び コンボボックス2としてカテゴリで絞り込んだ商品商品を入れる あとコンボボックスに無い内容でも 手入力で記入できるものを望んでいます。 (2)の回答 帳票形式です、 何か良いアイデアがあればお教え下さい。
お礼
piroin654 様 きめ細かな書き込みありがとう御座います、 頂いたご意見を元にじっくり整理しながら 作り上げたいと思います、 ネットにも同じような要望はあり、それに対して書き込みなどの アドバイスも色々御座いました、 帳票形式で出来ているサブフォーム内で、コンボボックスを複数つくり 絞り込み入力させる方法です。 ただそれらの情報の中で、コンボボックスから入力出来、なおかつコンボボックスに 無い物でも手動入力も出来る方法についてはネット上で探す事が出来ませんでした。 このようなニーズは以外に沢山ある気がするのですが。。。。 とりあえずpiroin654様の方法でがんばってみます 少々お時間をお掛け致しますが、結果報告いたします。
補足
piroin654 様 お世話になっております、 ご説明して頂いたようにaccessを作って見ましたが うまく行きません。 私が作成したaccessを以下URLに アップロードしております、何かご指摘して頂けると幸いです。 ファイル名:db10.mdb http://www.mediafire.com/?h7z2xbo68c1c6ct お気を悪くなさったら、申し訳ありません。 少々言い難いのですが、 もし宜しければpiroin654様が お作りになられた試作品を拝見出来れば 非常にあり難いのですが、、、、 気が向いたらで構いません、 私ももう一度、piroin654様のアドバイスを読み直して がんばってみます。