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

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

新規登録して質問してみよう
ただいま回答率
85.37%
Python 3.x

Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

Tkinter

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

ウィジェット

ウィジェットとはユーザインタフェイスの要素(GUI widget)であるか、もしくは、独立した比較的サイズの小さいソフトウェアアプリケーション(desktop widget)のことを指します。

Q&A

解決済

2回答

8987閲覧

tkinterで動的にウィジェットを生成・削除する方法について

nto

総合スコア1438

Python 3.x

Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

Tkinter

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

ウィジェット

ウィジェットとはユーザインタフェイスの要素(GUI widget)であるか、もしくは、独立した比較的サイズの小さいソフトウェアアプリケーション(desktop widget)のことを指します。

0グッド

0クリップ

投稿2020/03/03 04:41

tkinterで動的にウィジェットを生成・削除する方法について

GUI内でOptionMenuで指定した個数のウィジェットを生成したいです。
下記コードでは、はじめに個数(x)を選択後あとから個数(y)へ変更したとき
xよりyが下回る場合に対応する事ができません。

該当のソースコード

Python

1from tkinter import * 2 3 4#------------------------ 5class Application(Frame): 6 #-------------------------- 7 def __init__(self, master): 8 super().__init__(master) 9 self.grid() 10 11 master.geometry('300x120') 12 master.title('Test') 13 master.resizable(False, True) 14 15 self.widget1() 16 self.widget2() 17 18 def widget1(self): 19 Label(self.master, text='ID').grid(row=0, column=0, pady=5, padx=2) 20 uid_entry = Entry(self.master, width=8).grid(row=0, column=1) 21 22 def widget2(self): 23 Label(text='回数').grid(row=0, column=2) 24 self.var = IntVar() 25 self.var.set(1) 26 OptionMenu(self.master, self.var, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 27 command=self.select).grid(row=0, column=3) 28 Label(self.master, text='記入欄1').grid(row=1, column=1) 29 Label(self.master, text='記入欄2').grid(row=1, column=2) 30 31 def select(self,value): 32 i = value 33 self.widget_lst = [] 34 for w in range(i): 35 self.widget_lst.append(Entry(self.master, width=8)) 36 for n in range(i): 37 self.widget_lst[n].grid(row=n+2, column=1) 38 39def main(): 40 root = Tk() 41 app = Application(master = root) 42 app.mainloop() 43 44 45if __name__ == '__main__': 46 main()

試したこと

https://teratail.com/questions/144013
こちらの記事の回答者さんのサンプルコードを例にウィジェットの動的な生成はできましたが、あとからの個数変更に対応する事ができませんでした。
tryでwidget_lstの存在確認を行い、既に存在している場合にはリストを削除して生成し直す、存在していない場合にはそのまま生成する等試しましたがうまくいきませんでした。

うまくウィジェットの増減を動的に実現できるアドバイスをお願いします。

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

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

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

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

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

guest

回答2

0

grid_forget() にて Widgetを削除することができます。

Python

1from tkinter import * 2 3#------------------------ 4class Application(Frame): 5 #-------------------------- 6 def __init__(self, master): 7 super().__init__(master) 8 self.grid() 9 10 master.geometry('300x120') 11 master.title('Test') 12 master.resizable(False, True) 13 14 self.widget1() 15 self.widget2() 16 17 self.widget_lst = [] 18 19 def widget1(self): 20 Label(self.master, text='ID').grid(row=0, column=0, pady=5, padx=2) 21 uid_entry = Entry(self.master, width=8).grid(row=0, column=1) 22 23 def widget2(self): 24 Label(text='回数').grid(row=0, column=2) 25 self.var = IntVar() 26 OptionMenu(self.master, self.var, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 27 command=self.select).grid(row=0, column=3) 28 Label(self.master, text='記入欄1').grid(row=1, column=1) 29 Label(self.master, text='記入欄2').grid(row=1, column=2) 30 31 self.var.set(1) 32 33 def select(self,value): 34 new_value = value 35 cur_value = len(self.widget_lst) 36 if new_value > cur_value: 37 for p in range(new_value-cur_value): 38 entry = Entry(self.master, width=8) 39 entry.grid(row=cur_value+2+p, column=1) 40 self.widget_lst.append(entry) 41 elif new_value < cur_value: 42 for w in self.widget_lst[new_value:]: 43 w.grid_forget() 44 self.widget_lst = self.widget_lst[:new_value] 45 46def main(): 47 root = Tk() 48 app = Application(master = root) 49 app.mainloop() 50 51 52if __name__ == '__main__': 53 main()

投稿2020/03/03 06:06

magichan

総合スコア15898

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

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

nto

2020/03/03 07:14

回答ありがとうございます。 難しく考えてしまっておりました!
guest

0

ベストアンサー

python

1 2 def __init__(self, master): 3 super().__init__(master) 4 self.grid() 5 6 self.widget_lst = [] # 追加 7 8 9 def select(self,value): 10 i = value 11 for w in self.widget_lst: 12 w.pack_forget() 13 w.destroy() 14 self.widget_lst = [] 15 16

とりあえず、差分だけ。
destroyすれば良いです。上記の例では全部作り直しています。

投稿2020/03/03 06:03

t_obara

総合スコア5488

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.37%

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

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

質問する

関連した質問