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

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

ただいまの
回答率

89.10%

GUIカレンダーでクリックされた年月日を取得したい

解決済

回答 3

投稿 編集

  • 評価
  • クリップ 1
  • VIEW 1,247

3109

score 80

概要

下記サイトのPythonのtkinterで作成されたカレンダーを利用したいのですが、
クリックされた年月日の取得で躓いています。

python tkinter カレンダーの月めくり処理を実装する(カレンダー編③)
http://memopy.hatenadiary.jp/entry/2017/06/18/100406

memopy様、アップしていただきありがとうございます。
引用させていただきます。

実現したいこと

カレンダーの日付ボタンをクリックすると
クリックされたボタンのテキストを取得し、
トップに表示中の西暦と月を連結して
'20200112'のような8桁の年月日データを取り出したいと思っています。

(ゆくゆくはこのイベントで
選択された日の詳細データを取得する処理をさせる予定です。)

やってみたこと

クリックされたボタンのテキストを取得する関数を書き、
ボタンを作成するクラスとバインドさせることには成功したのですが、
トップに表示中の西暦と月の両テキストを取得するためには
どうしたらよいのかがわかりません。
イメージ説明
以下、私が新たに記述した関数です。

#左クリックされたカレンダーの年月日を取得する関数(作成中)
def callback(event):
    selected_date = ''
    if event.widget['text'] not in ['Mon','Tue','Wed','Thu','Fri','Sat','Sun']:
        selected_date += '2020'#ここに表示中の年を代入したい
        selected_date += '1'#ここに表示中の月を代入したい
        selected_date += str(event.widget['text'])
        print(selected_date)

スクリプト全文

以下、上記サイトから引用し手を加えたスクリプト全文です。

# -*- coding:utf-8 -*-

import tkinter as tk

# カレンダーを作成するフレームクラス
class mycalendar(tk.Frame):
    def __init__(self,master=None,cnf={},**kw):
        "初期化メソッド"
        import datetime
        tk.Frame.__init__(self,master,cnf,**kw)

        # 現在の日付を取得
        now = datetime.datetime.now()
        # 現在の年と月を属性に追加
        self.year = now.year
        self.month = now.month

        # frame_top部分の作成
        frame_top = tk.Frame(self)
        frame_top.pack(pady=5)
        self.previous_month = tk.Label(frame_top, text = "<", font = ("",14))
        self.previous_month.bind("<1>",self.change_month)
        self.previous_month.pack(side = "left", padx = 10)
        self.current_year = tk.Label(frame_top, text = self.year, font = ("",18))
        self.current_year.pack(side = "left")
        self.current_month = tk.Label(frame_top, text = self.month, font = ("",18))
        self.current_month.pack(side = "left")
        self.next_month = tk.Label(frame_top, text = ">", font = ("",14))
        self.next_month.bind("<1>",self.change_month)
        self.next_month.pack(side = "left", padx = 10)

        # frame_week部分の作成
        frame_week = tk.Frame(self)
        frame_week.pack()
        button_mon = d_button(frame_week, text = "Mon")
        button_mon.grid(column=0,row=0)
        button_tue = d_button(frame_week, text = "Tue")
        button_tue.grid(column=1,row=0)
        button_wed = d_button(frame_week, text = "Wed")
        button_wed.grid(column=2,row=0)
        button_thu = d_button(frame_week, text = "Thu")
        button_thu.grid(column=3,row=0)
        button_fri = d_button(frame_week, text = "Fri")
        button_fri.grid(column=4,row=0)
        button_sta = d_button(frame_week, text = "Sat", fg = "blue")
        button_sta.grid(column=5,row=0)
        button_san = d_button(frame_week, text = "Sun", fg = "red")#'San'→'Sun'と修正した
        button_san.grid(column=6,row=0)

        # frame_calendar部分の作成
        self.frame_calendar = tk.Frame(self)
        self.frame_calendar.pack()

        # 日付部分を作成するメソッドの呼び出し
        self.create_calendar(self.year,self.month)

    def create_calendar(self,year,month):
        "指定した年(year),月(month)のカレンダーウィジェットを作成する"

        # ボタンがある場合には削除する(初期化)
        try:
            for key,item in self.day.items():
                item.destroy()
        except:
            pass

        # calendarモジュールのインスタンスを作成
        import calendar
        cal = calendar.Calendar()
        # 指定した年月のカレンダーをリストで返す
        days = cal.monthdayscalendar(year,month)

        # 日付ボタンを格納する変数をdict型で作成
        self.day = {}
        # for文を用いて、日付ボタンを生成
        for i in range(0,42):
            c = i - (7 * int(i/7))
            r = int(i/7)
            try:
                # 日付が0でなかったら、ボタン作成
                if days[r][c] != 0:
                    self.day[i] = d_button(self.frame_calendar,text = days[r][c])
                    self.day[i].grid(column=c,row=r)
            except:
                """
                月によっては、i=41まで日付がないため、日付がないiのエラー回避が必要
                """
                break

    def change_month(self,event):
        # 押されたラベルを判定し、月の計算
        if event.widget["text"] == "<":
            self.month -= 1
        else:
            self.month += 1
        # 月が0、13になったときの処理
        if self.month == 0:
            self.year -= 1
            self.month = 12
        elif self.month == 13:
            self.year +=1
            self.month =1
        # frame_topにある年と月のラベルを変更する
        self.current_year["text"] = self.year
        self.current_month["text"] = self.month
        # 日付部分を作成するメソッドの呼び出し
        self.create_calendar(self.year,self.month)

# デフォルトのボタンクラス
class d_button(tk.Button):
    def __init__(self,master=None,cnf={},**kw):
        tk.Button.__init__(self,master,cnf,**kw)
        self.configure(font=("",14),height=2, width=4, relief="flat")
        self.bind('<Button-1>', callback)#オリジナルに追加した

#左クリックされたカレンダーの年月日を取得する関数(作成中)
def callback(event):
    selected_date = ''
    if event.widget['text'] not in ['Mon','Tue','Wed','Thu','Fri','Sat','Sun']:
        selected_date += '2020'#ここに表示中の年を代入したい
        selected_date += '1'#ここに表示中の月を代入したい
        selected_date += str(event.widget['text'])
        print(selected_date)

# ルートフレームの定義      
root = tk.Tk()
root.title("Calendar App")
mycal = mycalendar(root)
mycal.pack()
root.mainloop()

補足

current_year["text"]current_month["text"]
目的のデータを格納していると思うのですが、
私はクラスに関して初心者のためどう扱ったらよいのかわかりません。

さらに私が新規で書いた関数についても記述する場所はここでよいのか、
それともどちらかのクラスのメソッドとした方がよいのか、よく理解できていません。
その点もアドバイスをいただけると幸いです。

お知恵をお貸しください。よろしくお願いします。

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

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

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

    クリップを取り消します

  • 良い質問の評価を上げる

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

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

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

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

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

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

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

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 3

check解決した方法

0

自己解決しました。
正確に言うと、目的実現のためにアプローチ法を変えました。

「GUIに表示されている年・月を取得する」ことを諦めて、
「年・月の表示が更新される度にグローバル変数を上書きし、
各日ボタン押下時にそれらの年・月・日を結合する」ことにしました。

以下スクリプト全文です。
私が付け足した部分及びブロックには
「追記」とコメントアウトしてあります。

# -*- coding:utf-8 -*-

import tkinter as tk

# カレンダーを作成するフレームクラス
class mycalendar(tk.Frame):
    def __init__(self,master=None,cnf={},**kw):
        "初期化メソッド"
        import datetime
        tk.Frame.__init__(self,master,cnf,**kw)

        # 現在の日付を取得
        now = datetime.datetime.now()
        # 現在の年と月を属性に追加
        self.year = now.year
        self.month = now.month
        # 追記 https://teratail.com/questions/234639#reply-355304
        global YEAR, MONTH
        YEAR = str(self.year)
        MONTH = str(self.month)

        # frame_top部分の作成
        frame_top = tk.Frame(self)
        frame_top.pack(pady=5)
        self.previous_month = tk.Label(frame_top, text = "<", font = ("",14))
        self.previous_month.bind("<1>",self.change_month)
        self.previous_month.pack(side = "left", padx = 10)
        self.current_year = tk.Label(frame_top, text = self.year, font = ("",18))
        self.current_year.pack(side = "left")
        self.current_month = tk.Label(frame_top, text = self.month, font = ("",18))
        self.current_month.pack(side = "left")
        self.next_month = tk.Label(frame_top, text = ">", font = ("",14))
        self.next_month.bind("<1>",self.change_month)
        self.next_month.pack(side = "left", padx = 10)

        # frame_week部分の作成
        frame_week = tk.Frame(self)
        frame_week.pack()
        button_mon = d_button(frame_week, text = "Mon")
        button_mon.grid(column=0,row=0)
        button_tue = d_button(frame_week, text = "Tue")
        button_tue.grid(column=1,row=0)
        button_wed = d_button(frame_week, text = "Wed")
        button_wed.grid(column=2,row=0)
        button_thu = d_button(frame_week, text = "Thu")
        button_thu.grid(column=3,row=0)
        button_fri = d_button(frame_week, text = "Fri")
        button_fri.grid(column=4,row=0)
        button_sta = d_button(frame_week, text = "Sat", fg = "blue")
        button_sta.grid(column=5,row=0)
        button_san = d_button(frame_week, text = "Sun", fg = "red")#'San'→'Sun'と修正した
        button_san.grid(column=6,row=0)

        # frame_calendar部分の作成
        self.frame_calendar = tk.Frame(self)
        self.frame_calendar.pack()

        # 日付部分を作成するメソッドの呼び出し
        self.create_calendar(self.year,self.month)

    def create_calendar(self,year,month):
        "指定した年(year),月(month)のカレンダーウィジェットを作成する"

        # ボタンがある場合には削除する(初期化)
        try:
            for key,item in self.day.items():
                item.destroy()
        except:
            pass

        # calendarモジュールのインスタンスを作成
        import calendar
        cal = calendar.Calendar()
        # 指定した年月のカレンダーをリストで返す
        days = cal.monthdayscalendar(year,month)

        # 日付ボタンを格納する変数をdict型で作成
        self.day = {}
        # for文を用いて、日付ボタンを生成
        for i in range(0,42):
            c = i - (7 * int(i/7))
            r = int(i/7)
            try:
                # 日付が0でなかったら、ボタン作成
                if days[r][c] != 0:
                    self.day[i] = d_button(self.frame_calendar,text = days[r][c])
                    self.day[i].grid(column=c,row=r)
            except:
                """
                月によっては、i=41まで日付がないため、日付がないiのエラー回避が必要
                """
                break

    def change_month(self,event):
        # 押されたラベルを判定し、月の計算
        if event.widget["text"] == "<":
            self.month -= 1
        else:
            self.month += 1
        # 月が0、13になったときの処理
        if self.month == 0:
            self.year -= 1
            self.month = 12
        elif self.month == 13:
            self.year +=1
            self.month =1
        # frame_topにある年と月のラベルを変更する
        self.current_year["text"] = self.year
        self.current_month["text"] = self.month

        # 追記 https://teratail.com/questions/234639#reply-355304
        global YEAR, MONTH
        YEAR = str(self.year)
        MONTH = str(self.month)

        # 日付部分を作成するメソッドの呼び出し
        self.create_calendar(self.year,self.month)

# デフォルトのボタンクラス
class d_button(tk.Button):
    def __init__(self,master=None,cnf={},**kw):
        tk.Button.__init__(self,master,cnf,**kw)
        self.configure(font=("",14),height=2, width=4, relief="flat")
        self.bind('<Button-1>', callback)# 追記 https://teratail.com/questions/234639

# カレンダーの年月日を取得するコールバック関数
# 追記 https://teratail.com/questions/234639#reply-355304
def callback(event):
    selected_date = ''
    if event.widget['text'] not in ['Mon','Tue','Wed','Thu','Fri','Sat','Sun']:
        selected_date += YEAR
        selected_date += convert_in2_2bytes(MONTH)
        selected_date += convert_in2_2bytes(str(event.widget['text']))
        print(selected_date)

# 1桁の数字を2バイトに変換する関数
# 追記 https://teratail.com/questions/234639#reply-355304
def convert_in2_2bytes(str_number):
    if len(str_number) == 1:
        return '0' + str_number
    else:
        return str_number

# ルートフレームの定義      
root = tk.Tk()
root.title("Calendar App")
mycal = mycalendar(root)
mycal.pack()
root.mainloop()


メインループ中にグローバル変数を宣言しなくても
動作するのを確認したので、明示はしませんでした。
もしも作法として誤りであれば御指摘ねがいます。

datetimeモジュールが吐き出す
年・月・日の桁数の検査は記述していません。
フル・フロンタル風に言うならば、
「もし、西暦がカンストしたとしたら、
それはもう、Pythonではなくなっているのではないかな。」
ということで。

もっとスマートで省メモリな書き方があれば、
ご教示ください。よろしくお願いします。

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

0

selected_date += '2020'#ここに表示中の年を代入したい


selected_date += current_year["text"] #年のラベルの値を使う


にしては、どうでしょう?

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2020/01/12 12:33

    coco_bauerさん、回答ありがとうございます。

    試してみたのですが下記のようにエラーとなります。

    selected_date += current_year["text"]
    NameError: name 'current_year' is not defined

    やはり'current_year'がどこに属しているのかわからないので、
    認識してくれません。

    キャンセル

0

3109さん
私も丁度同じようなことで色々調べていたので大変参考になりました。ありがとうございます。

ところで、この方法だとコードを実行したときに今月の日付を取得するには、
一度月のボタンをクリックし月を変化させないと年と月を取得できないのではないでしょうか?
そこで、私は「カレンダーの年月日を取得するコールバック関数」の部分を以下のようにしてみました。

def callback(event):
    selected_date = ''
    if event.widget['text'] not in ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun']:
        try:
            selected_date += YEAR
        except:
            selected_date += str(now.year)
        selected_date += '/'
        try:
            selected_date += convert_in2_2bytes(MONTH)
        except:
            selected_date += convert_in2_2bytes(str(now.month))
        selected_date += '/'
        selected_date += convert_in2_2bytes(str(event.widget['text']))
        print(selected_date)

年と月が取得できないときには、今日の年月を取得することができました。
年月を分ける / は、この後私が使いやすい形式ししただけですが

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2020/03/12 09:08

    Q_1986-ktさん、回答ありがとうございます。

    私もボタン押下時しか年月が取得できないことには気づいていましたが、
    私のシステムでは年月が必要な時は必ずボタンを通過する設計になっていますので、
    記述はしませんでした。'/'区切りについても同様の考えです。

    実現したいことは人それぞれでしょうから、
    これからここを訪れる迷い人の助けになるかもしれませんね。
    ありがとうございます。

    キャンセル

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

  • ただいまの回答率 89.10%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

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