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

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

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

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

Python

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

Q&A

2回答

929閲覧

tkinterを使用してストップウォッチの作成

yu_jin

総合スコア15

Tkinter

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

Python

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

0グッド

0クリップ

投稿2022/08/28 13:47

編集2022/08/28 13:53

前提

Pythonでtkinterを使用し、ストップウォッチ作成しています。
タイマー用のキャンパスorラベルに計測結果の数値を表示させたいのですが、表示されません。

タイマー用のキャンパスorラベルがmain_frameに作成されていて、計測結果が別のところに表示されているのではないかと思っているのですが、その問題を解決する方法がわかりません。

初歩的な質問だと思いますが、教えて頂きたく思います。
よろしくお願い致します。

実現したいこと

プログラム実行時に、ラベル内に0.0が表示される。
スタート、ストップ、リセットのボタンを押したらその結果をタイマー用のキャンパスorラベルに表示させたい。

試したこと

ラベルで作成した際に、text = updateでupdateの内容を表示させるようにした。
参考にしたコードを書いて実行した。

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

windows10 Python3.9

発生している問題・エラーメッセージ

エラーメッセージ

該当のソースコード

コード from ast import Lambda import tkinter as tk import PIL.Image, PIL.ImageTk import datetime import time root = tk.Tk() #----------------計測に使う数値-------------------------------------------- #-----------------------------------クラス宣言---------------------------- #tk.Frame = テキストやボタンなどを埋め込む四角い領域を作るためのコマンド class Stopwatch_frame(tk.Frame): #class tk.Frameを継承してStopwatchを定義 def __init__(self,main_Frame): #コンストラクタ super().__init__(main_Frame) #親クラスのメソッドを実行 # ウィンドウタイトルを決定 self.master.title("ストップウォッチ") # ウィンドウの大きさを決定 self.master.geometry("800x600") self.master.config(bg="black") self.startTime=time.time() self.stopTime=0.0 self.elapsedTime=0.0 self.playTime=False # ウィンドウのグリッドを 1x1 にする self.master.grid_rowconfigure(0, weight=1) self.master.grid_columnconfigure(0, weight=1) #-----------------------------------main_frame----------------------------- # メインページフレーム作成 self.main_frame = tk.Frame(main_Frame, bg='white') self.main_frame.grid(row=0, column=0, sticky="nsew") #日付ラベルの作成 self.label = tk.Label(self.main_frame,text=datetime.date.today(),font=("", 30, "bold"),background='skyblue') self.label.place(x=300,y=100) #-----------------------------------移動先frame--------------------------------- # 移動先フレーム作成 self.sub_frame = tk.Frame(bg='white') self.sub_frame.grid(row=0, column=0, sticky="nsew") # タイトルラベル作成 self.titleLabel = tk.Label(self.sub_frame, text="Frame 1", font=('Helvetica', '35')) self.titleLabel.place(x = 400, y = 400) #main_frameを一番上に表示 self.main_frame.tkraise() #-----------------------------------main_frame部品----------------------------- #メインフレームボタン self.resetButton = tk.Button(self.main_frame,text="リセット", command=lambda: resetButtonClick(self), width=20).place(x=550, y=370) self.startButto = tk.Button(self.main_frame,text="スタート", command=lambda: startButtonClick(self), width=20).place(x=100, y=370) self.stopButton = tk.Button(self.main_frame,text="ストップ", command=lambda: stopButtonClick(self), width=20).place(x=320, y=370) # self.changePageButton = tk.Button(self.main_frame, text="次のページ", width=20).place(x=550, y=450) self.changePageButton = tk.Button(self.main_frame, text="次のページ", command=lambda: change_page(self.sub_frame), width=20).place(x=550, y=450) self.recordButton = tk.Button(self.main_frame,text="記録",width=20).place(x=320, y=450) #タイマー用のキャンパスorラベル #self.canvas = tk.Canvas(self.main_frame,width=400,height=120,bg="skyblue") #self.canvas.place(x=200,y=180) #self.timer_label = tk.Label(self.main_frame,text=update(self),width=60,height=7,bg="skyblue") #self.timer_label.place(x=200,y=180) self.canvas = tk.Canvas(self.main_frame,width=290,height=80,bg="skyblue") self.canvas.place(x=3,y=10) self.main_frame.after(50,self.update) #----------------関数----------------------------------------------------- #ページの切り替え def change_page(page): page.tkraise() def resetButtonClick(self): self.startTime=time.time() self.stopTime=0.0 self.elapsedTime=0.0 self.playTime=False #計測をやめる def startButtonClick(self): if not self.playTime: #flagがFalseだったら実行(計測を行っていない) self.startTime=time.time()-self.elapsedTime #スタート時間から経過時間を引いた値を格納 self.playTime=True #flagをTrueに変える(計測を行う) def stopButtonClick(self): if self.playTime: self.stopTime=time.time()-self.startTime #現在の時刻から開始時間を引いた値を格納 self.playTime=False #flagをTrueに変える(計測を行っていない) def update(self): self.main_frame.canvas.delete("Time") if self.playTime: self.elapsedTime=time.time()-self.startTime self.main_frame.canvas.create_text(280,40,text=round(self.elapsedTime,1),font=("Helvetica",40,"bold"),fill="black",tag="Time",anchor="e") else: self.main_frame.canvas.create_text(280,40,text=round(self.stopTime,1),font=("Helvetica",40,"bold"),fill="black",tag="Time",anchor="e") self.main_frame.after(50,self.update) """def update(self): self.canvas.delete("Time") if self.playTime == True: self.elapsedTime=time.time()-self.startTime self.canvas.create_text(200,180,text=round(self.elapsedTime,1),font=("Helvetica",40,"bold"),fill="black",tag="Time",anchor="e") else: #round関数で第二引数は小数点で丸める桁の指定 self.canvas.create_text(200,180,text=round(self.stopTime,1),font=("Helvetica",40,"bold"),fill="black",tag="Time",anchor="e") self.master.after(50,self.update) #更新させる間隔 """ #ラベルで試したもの """def update(self): self.master.timer_label.delete("Time") if self.playTime == True: self.elapsedTime=time.time()-self.startTime self.timer_label.create_text(200,180,text=round(self.elapsedTime,1),font=("Helvetica",40,"bold"),fill="black",tag="Time",anchor="e") else: #round関数で第二引数は小数点で丸める桁の指定 self.timer_label.create_text(200,180,text=round(self.stopTime,1),font=("Helvetica",40,"bold"),fill="black",tag="Time",anchor="e") self.after(50,self.update) #更新させる間隔 """ def main(): app = Stopwatch_frame(root) app.mainloop() if __name__ == "__main__": ## モジュールを直接実行した時だけ、実行したいコード main()

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

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

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

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

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

guest

回答2

0

ソースコードのレイアウトを見直しましょう。
クラスのメソッドとして定義したいはずの updateが、トップレベルの関数として実装されています。

tkinter の Widget は、全て update メソッドを持っているので、
tk.Frame を継承した Stopwatch_frame クラスからは、親クラスのメソッドを呼び出していて
自分で実装した update() が使われていません。

メソッドとして扱われるように、適切な位置にインデントを挿入してください。

目的の挙動を実装するためには他にも、start, stop ボタンにも同様の問題がありますが、
上記の修正を行うことでエラーメッセージが表示されるようになるはずなので、
先に進めるはずです。

他の問題点

  • self.main_frame.canvas は未定義。 Pythonの属性で参照する場合は self.canvas

 (tk内部での扱いでは .frame.canvas のような、親から列挙する階層表記になります)

  • self.button = tk.Button(...).place(...) self.button には None が入るので後からボタンを参照できない

注意したほうが良い点
tkinter 関連のクラスでは、ウィジェットを継承する場合は
"update" という名前でのメソッドの上書きは避けたほうが良いです。
(コード的には正しくても、親クラスのメソッドと混同されることがあるため)

投稿2022/08/29 05:04

編集2022/08/29 05:06
teamikl

総合スコア8664

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

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

0

afterのタイマーが50msなのは短かすぎるように思うので、これを、10秒=10000msくらいにして試してみたらどうでしょう。

また、表示の更新ができない原因を切り分けるために、1ずつ増える数字などもう少し簡単なものを表示してみてはいかがでしょうか。

投稿2022/08/28 14:26

TakaiY

総合スコア12765

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

まだベストアンサーが選ばれていません

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

アカウントをお持ちの方は

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問