質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
Tkinter

Tkinterは、GUIツールキットである“Tk”をPythonから利用できるようにした標準ライブラリである。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

Q&A

解決済

1回答

1280閲覧

【tkinter】マウススクロール時、コンボボックスのプルダウンリストをしまいたい【Python】

netz-eng

総合スコア105

Tkinter

Tkinterは、GUIツールキットである“Tk”をPythonから利用できるようにした標準ライブラリである。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

0グッド

0クリップ

投稿2022/06/27 14:04

実現したいこと

独学でGUIを開発しております。

マウススクロールをバインドしたフレーム内にコンボボックスを配置し、

コンボボックスのプルダウンリストを表示した状態で、
コンボボックスの外(フレーム部分)をマウススクロールしたとき、
プルダウンリストをしまいたいのですが、コードの書き方がわかりません。

現状では、プルダウンリストがある状態でコンボボックスの外でマウススクロールをすると、
プルダウンリストが表示されたまま画面がスクロールされてしまいます。

マウススクロール時に呼び出される関数のなかで、winfo_existspopdownの存在を判定すればよいと思うのですが、
ループでコンボボックスを作成しているため、ウィジェット変数名がなく判定ができません。

解決方法にお心当たりがあれば、ご回答よろしくお願いします。

該当のソースコード

Python

1import tkinter as tk 2import tkinter.ttk as ttk 3 4app = tk.Tk() 5fra_outcan = tk.Frame(app) 6fra_outcan.grid(row=1, column=0) 7# can = tk.Canvas(app) 8can = tk.Canvas(fra_outcan, highlightthickness=0) 9 10ybar = ttk.Scrollbar(fra_outcan, orient=tk.VERTICAL, command=can.yview, takefocus=0) 11ybar.grid(row=0, column=1, sticky=tk.N+tk.S) 12can.configure(yscrollcommand=ybar.set) 13can.grid(row=0, column=0, pady=10) 14 15def MouseScroll(event): 16 w = event.widget 17 sw = str(w) 18 #プルダウンリスト上でマウススクロールした場合は、リスト内をスクロールする 19 if not "popdown" in sw: 20 can.yview("scroll", int(-1*(event.delta/120)), "units") 21 22fra_incan = tk.Frame(can, height=50) 23can.create_window((0, 0), window=fra_incan, anchor=tk.NW, width=can.cget('width')) 24 25fra_incan.bind_all("<MouseWheel>", MouseScroll) 26 27texs = [] 28lst_cb = list(range(20)) 29for i in range(20): 30 tex = ttk.Combobox(fra_incan, width=50, values=lst_cb) 31 texs.append(tex) 32 tex.set(i) 33 # すべてのコンボボックスからマウスホイールを除去 34 app.unbind_class("TCombobox", "<MouseWheel>") 35 # マウススクロール関数をバインド 36 texs[i].bind_all("<MouseWheel>", MouseScroll) 37 texs[i].pack(pady=5) 38 39app.update_idletasks() 40x = (texs[i].winfo_height()+10)*10 41can.config(scrollregion=(0, 0, 100, x)) 42can.config(height=150) 43fra_outcan.config(relief=tk.RAISED, borderwidth=5) 44print(str(x)) 45 46ybar.configure(style="c.Vertical.TScrollbar") 47 48app.mainloop()

補足情報(FW/ツールのバージョンなど)

Windows11
Python 3.9.7
tk 8.6.11

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

コンボボックスのプルダウンリストを表示した状態で、

表示されているかどうかの判別は、前回の回答を参照してください。
winfo exists と winfo viewable で、生成されているか・表示されているかを2重確認します。

コンボボックスの外(フレーム部分)をマウススクロールしたとき、

特定ウィジェット外の判別は、コンボボックスウィジェット側のbindでは把握できないので、
bind_all で全てのスクロールイベントを所得して、その中で対象のイベントかどうかを判別します。

python

1def onWheel(event): 2 widget = event.widget 3 ... 4 5root.bind_all("<MouseWheel>", onWheel)

コンボボックスのポップダウンは内部ウィジェットなので
ウィジェットのIDは print(event.widget) 等で調べて見て下さい。

プルダウンリストをしまいたいのですが、コードの書き方がわかりません。

プルダウンリストを仕舞うのはPython 側のメソッドとしては提供されていないので、
tcl eval 経由で ttk::combobox::Unpost を呼び出します。

python

1root.tk.eval(f"ttk::combobox::Unpost {combo}")

投稿2022/06/28 02:58

編集2022/06/28 03:44
teamikl

総合スコア8664

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

netz-eng

2022/06/29 12:42

前回に引き続き、ご回答ありがとうございます。 >コンボボックスのポップダウンは内部ウィジェットなので >ウィジェットのIDは print(event.widget) 等で調べて見て下さい。 このあたりの操作で困っていたのですが、マウススクロール関数のなかでコンボボックスのリストを走査して、 ポップダウンがあればしまう、という処理をすることで解決しました。 難しく考えすぎていたようです。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問