回答編集履歴

3 リンク・アドレスの修正

3109

3109 score 80

2020/03/12 08:42  投稿

自己解決しました。
正確に言うと、目的実現のためにアプローチ法を変えました。
「GUIに表示されている年・月を取得する」ことを諦めて、
「年・月の表示が更新される度にグローバル変数を上書きし、
各日ボタン押下時にそれらの年・月・日を結合する」ことにしました。
以下スクリプト全文です。
私が付け足した部分及びブロックには
「追記」とコメントアウトしてあります。
```Python
# -*- 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
       # 追記 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
       # 追記 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
# 追記 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
# 追記 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ではなくなっているのではないかな。」
ということで。
もっとスマートで省メモリな書き方があれば、
ご教示ください。よろしくお願いします。
2 表現の最適化

3109

3109 score 80

2020/02/02 12:44  投稿

自己解決しました。
正確に言うと、目的実現のためにアプローチ法を変えました。
「GUIに表示されている年・月を取得する」ことを諦めて、
「年・月の表示が更新される度にグローバル変数を上書きし、
各日ボタン押下時にそれらの年・月・日を結合する」ことにしました。
以下スクリプト全文です。
私が付け足した部分及びブロックには
「追記」とコメントアウトしてあります。
```Python
# -*- 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
       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
       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
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
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ではなくなっているのではないかな。」
ということで。
もっとスマートで省メモリな書き方があれば、
ご教示ください。よろしくお願いします。
1 モジュール名の訂正

3109

3109 score 80

2020/02/01 14:15  投稿

自己解決しました。
正確に言うと、目的実現のためにアプローチ法を変えました。
「GUIに表示されている年・月を取得する」ことを諦めて、
「年・月の表示が更新される度にグローバル変数を上書きし、
各日ボタン押下時にそれらの年・月・日を結合する」ことにしました。
以下スクリプト全文です。
私が付け足した部分及びブロックには
「追記」とコメントアウトしてあります。
```Python
# -*- 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
       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
       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
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
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()
```
メインのループ中にグローバル変数を宣言しなくても
動作したので明示はしませんでした。
もしも作法として誤りであれば御指摘ねがいます。
calendarモジュールが吐き出す
datetimeモジュールが吐き出す
年・月・日の桁数の検査は記述していません。
フル・フロンタル風に言うならば、
「もし、西暦がカンストしたとしたら、
それはもう、Pythonではなくなっているのではないかな。」
ということで。
もっとスマートで省メモリな書き方があれば、
ご教示ください。よろしくお願いします。

思考するエンジニアのためのQ&Aサイト「teratail」について詳しく知る