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

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

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

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

Tkinter

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

解決済

Python3 Tkinter 時刻入力専用画面の作成

person
person

総合スコア203

Python 3.x

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

Tkinter

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

1回答

0評価

0クリップ

547閲覧

投稿2022/04/11 06:39

編集2022/04/14 16:19

Tkinterで時刻入力専用画面を作成します。

Python3 Tkinter 時間入力やテンキーなどの専用画面をウィジェットのすぐ下に表示したい
の回答を参考に、表示・非表示を切り替えるように作成しました。

仕様として
Treeviewレコードの時刻データ列をダブルクリックすると時刻入力画面が出て、時刻の変更が可能というものです。

時刻入力画面の左上の座標が、該当時刻列の左下。
ただし、①ディスプレイ上より右にはみ出る場合は、入力画面の左端を該当時刻列の右端に揃える。
②ディスプレイ上より下にはみ出る場合(タスクバーは考慮しない)は、入力画面の下端を該当時刻列の上端に揃える。
というふうにしたいです。

作ってみたのですが、
①は、惜しいような気もしますが目標より若干右よりなきがします。
②も目標より下にずれている感じです。
単に補正が必要なのか計算が間違っているのか分かりません。
分かる方教えていただけると幸いです。

(座標計算はpopup_time_input.pyのTimeInputクラスのcall_time_input関数でやってます)

また、他に改善点あればそちらもご指摘お願いします。

目標位置と実際の位置の差
黄:目標位置
緑:実際のウィンドウの端の位置
イメージ説明

Python

# main.py <- entry point import tkinter as tk from tkinter import ttk import popup_time_input class App(tk.Frame): def __init__(self, master): super().__init__(master) # tk.Frame(root) に相当 self.time_input = popup_time_input.TimeInput() self.create_widgets() def create_widgets(self): self.tree = ttk.Treeview(root, selectmode="browse") self.tree.pack() # headings columns = 2 self.tree["columns"] = tuple(range(columns)) self.tree["show"] = "headings" self.tree.heading(0, text="No.") self.tree.heading(1, text="Time[hh:mm]") # data self.tree.insert("", "end", values=("1", "00:00")) self.tree.insert("", "end", values=("2", "01:11")) self.tree.insert("", "end", values=("3", "02:22")) self.tree.insert("", "end", values=("4", "03:33")) self.tree.insert("", "end", values=("5", "04:44")) self.tree.bind("<Double-Button-1>", self.on_tree) def on_tree(self, e): tree = e.widget selected = tree.selection() # 選択行 clickarea = tree.identify_region(e.x, e.y) # "heading" or "cell" if len(selected) == 1 and clickarea == "cell": # 列取得 x = tree.winfo_pointerx() - tree.winfo_rootx() select_column_str = tree.identify_column(x) select_column_int = int(select_column_str[1:]) # 時刻列ダブルクリックで専用入力画面表示 if select_column_int == 2: self.time_input.call_time_input(tree, select_column_int) if __name__ == "__main__": root = tk.Tk() app = App(root) app.mainloop()

Python

#popup_time_input.py import tkinter as tk from tkinter import ttk class TimeInput: def __init__(self): self.width = 400 self.height = 300 self.tree = None self.tree_iid = None self.select_column = None self.input_window = tk.Toplevel() self.input_window.withdraw() self.input_window.title("InputWindow") self.create_widgets() self.input_window.protocol("WM_DELETE_WINDOW", lambda:None) def create_widgets(self): self.input_window.rowconfigure((0,1,2,3), weight=1) self.input_window.columnconfigure((0,2), weight=1) self.hour = tk.StringVar() hour_entry = ttk.Entry(self.input_window, textvariable=self.hour) hour_entry.grid(row=0, column=0) colon = ttk.Label(self.input_window, text=":") colon.grid(row=0, column=1) self.minu = tk.StringVar() minu_entry = ttk.Entry(self.input_window, textvariable=self.minu) minu_entry.grid(row=0, column=2) hourplus = ttk.Button(self.input_window, text="+", command=self.on_hourplus) hourplus.grid(row=1, column=0, sticky="nsew") minuplus = ttk.Button(self.input_window, text="+", command=self.on_minuplus) minuplus.grid(row=1, column=2, sticky="nsew") hourminus = ttk.Button(self.input_window, text="-", command=self.on_hourminus) hourminus.grid(row=2, column=0, sticky="nsew") minuminus = ttk.Button(self.input_window, text="-", command=self.on_minuminus) minuminus.grid(row=2, column=2, sticky="nsew") ok = ttk.Button(self.input_window, text="OK", command=self.on_ok) ok.grid(row=3, column=0) cancel = ttk.Button(self.input_window, text="cancel", command=self.on_cancel) cancel.grid(row=3, column=2) def on_hourplus(self): # hour + txt = self.hour.get() try: if int(txt) < 23: txt = str((int(txt) + 1)).zfill(2) else: txt = "00" self.hour.set(txt) except: pass def on_minuplus(self): # minutes + txt = self.minu.get() try: if int(txt) < 59: txt = str((int(txt) + 1)).zfill(2) else: txt = "00" self.minu.set(txt) except: pass def on_hourminus(self): # hour - - txt = self.hour.get() try: if int(txt) > 0: txt = str((int(txt) - 1)).zfill(2) else: txt = "23" self.hour.set(txt) except: pass def on_minuminus(self): # minutes - txt = self.minu.get() try: if int(txt) > 0: txt = str((int(txt) - 1)).zfill(2) else: txt = "59" self.minu.set(txt) except: pass def on_ok(self): children = self.tree.get_children() new_data = self.hour.get() + ":" + self.minu.get() self.tree.set(self.tree_iid, self.select_column - 1, new_data) self.input_window.grab_release() self.input_window.withdraw() def on_cancel(self): self.input_window.grab_release() self.input_window.withdraw() def call_time_input(self, tree, select_column): # 表示用関数(外部から呼ぶ) self.tree = tree self.select_column = select_column is_select = tree.selection() if is_select: self.tree_iid = is_select[0] value = tree.item(self.tree_iid)["values"][select_column - 1] print("value: {}".format(value)) self.hour.set(value[0:2]) self.minu.set(value[3:5]) # Treeview該当レコードの左下にウィンドウの左上を設定 x, y, w, h = tree.bbox(self.tree_iid, select_column - 1) x += tree.winfo_rootx() y += tree.winfo_rooty() + h print(y) print(h) print(self.input_window.winfo_screenheight()) print(self.input_window.winfo_height()) # はみ出すようなら、Treeview該当レコードの右上にウィンドウの右下を設定 limitx = self.input_window.winfo_screenwidth() - self.input_window.winfo_width() limity = self.input_window.winfo_screenheight() - self.input_window.winfo_height() if x >= limitx: x = x - self.input_window.winfo_width() if y >= limity: y = y - h - self.input_window.winfo_height() geometry = f"{self.width}x{self.height}+{x}+{y}" print("geometry: {}".format(geometry)) self.input_window.geometry(geometry) self.input_window.deiconify() self.input_window.grab_set()

良い質問の評価を上げる

以下のような質問は評価を上げましょう

  • 質問内容が明確
  • 自分も答えを知りたい
  • 質問者以外のユーザにも役立つ

評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

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

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

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

teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

  • プログラミングに関係のない質問
  • やってほしいことだけを記載した丸投げの質問
  • 問題・課題が含まれていない質問
  • 意図的に内容が抹消された質問
  • 過去に投稿した質問と同じ内容の質問
  • 広告と受け取られるような投稿

評価を下げると、トップページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

まだ回答がついていません

会員登録して回答してみよう

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

ただいまの回答率
87.20%

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

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

質問する

関連した質問

同じタグがついた質問を見る

Python 3.x

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

Tkinter

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