-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.py
192 lines (158 loc) · 6.29 KB
/
main.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
import tkinter as tk
import json, os, uuid, sys, winreg
from datetime import datetime
from pathlib import Path
from event_frame import EventFrame
def add_to_startup():
"""
컴퓨터 부팅 시 자동으로 실행되도록 레지스트리에 등록.
콘솔창이 뜨지 않도록 pythonw.exe 사용.
이미 등록되어 있으면 등록 과정을 스킵합니다.
"""
try:
script_path = os.path.abspath(sys.argv[0])
executable = sys.executable
# 만약 sys.executable이 "python.exe"라면 "pythonw.exe"로 변경
if "python.exe" in executable.lower():
executable = executable.replace("python.exe", "pythonw.exe")
command = f'"{executable}" "{script_path}"'
reg_path = r"Software\Microsoft\Windows\CurrentVersion\Run"
key_name = "MyTkinterApp"
# 먼저 읽기 권한으로 키를 열어 등록 여부 확인
try:
reg_key = winreg.OpenKey(winreg.HKEY_CURRENT_USER, reg_path, 0, winreg.KEY_READ)
try:
current_value, _ = winreg.QueryValueEx(reg_key, key_name)
if current_value == command:
print("이미 등록되어 있습니다.")
winreg.CloseKey(reg_key)
return
except FileNotFoundError:
# 등록된 값이 없으면 FileNotFoundError가 발생하므로 무시
pass
winreg.CloseKey(reg_key)
except Exception as e:
# 읽기 권한으로 열 때 문제가 발생하면 그대로 진행
print("등록 확인 중 문제 발생:", e)
# 등록되어 있지 않으므로 쓰기 권한으로 키를 열어 값을 설정
reg_key = winreg.OpenKey(winreg.HKEY_CURRENT_USER, reg_path, 0, winreg.KEY_SET_VALUE)
winreg.SetValueEx(reg_key, key_name, 0, winreg.REG_SZ, command)
winreg.CloseKey(reg_key)
print("자동 실행 등록 성공!")
except Exception as e:
print("자동 실행 등록 실패:", e)
# 필요하다면 주석 해제하여 부팅 시 자동 실행 등록
# add_to_startup()
# 사용자 홈 디렉터리 내 .my_d_day_widget 폴더 생성
user_data_dir = Path(os.path.expanduser('~')) / ".my_d_day_widget"
user_data_dir.mkdir(parents=True, exist_ok=True)
# JSON 파일을 홈 디렉터리 하위 폴더에 저장
EVENTS_FILE = user_data_dir / "events.json"
def load_events():
"""JSON 파일에서 일정 정보를 불러옴 (없으면 기본 일정 생성)"""
if EVENTS_FILE.exists():
with open(EVENTS_FILE, "r", encoding="utf-8") as f:
events = json.load(f)
# target_date 문자열 -> datetime 변환
for event in events:
event["target_date"] = datetime.strptime(event["target_date"], "%Y-%m-%d %H:%M")
return events
else:
# 기본 일정 데이터
return [
{"id": str(uuid.uuid4()), "title": "일정 1", "target_date": datetime(2000, 1, 1, 0, 0)},
{"id": str(uuid.uuid4()), "title": "일정 2", "target_date": datetime(2025, 2, 25, 0, 0)},
{"id": str(uuid.uuid4()), "title": "일정 3", "target_date": datetime(2030, 1, 1, 0, 0)},
]
def save_events():
"""event_data를 JSON 파일에 저장"""
to_save = []
for event in event_data:
to_save.append({
"id": event["id"],
"title": event["title"],
"target_date": event["target_date"].strftime("%Y-%m-%d %H:%M")
})
with open(EVENTS_FILE, "w", encoding="utf-8") as f:
json.dump(to_save, f, ensure_ascii=False, indent=4)
def update_all():
"""전체 일정 UI 갱신 및 재배치"""
for widget in container_frame.winfo_children():
widget.destroy()
event_frames.clear()
# 날짜 순으로 정렬하여 EventFrame 생성
sorted_data = sorted(event_data, key=lambda d: d["target_date"])
for data in sorted_data:
ef = EventFrame(
container_frame,
data["title"],
data["target_date"],
update_callback=update_all,
add_event_callback=add_event,
update_event_callback=update_event,
delete_event_callback=delete_event,
event_id=data["id"],
width=220,
height=120,
bg="gray"
)
ef.pack(side="left", padx=10)
event_frames.append(ef)
# 메인 창 크기 재설정
root.update_idletasks()
win_width = container_frame.winfo_reqwidth() + 20
win_height = container_frame.winfo_reqheight() + 20
screen_width = root.winfo_screenwidth()
x_pos = screen_width - win_width - 10
y_pos = 50
root.geometry(f"{win_width}x{win_height}+{x_pos}+{y_pos}")
# 1분마다 갱신
root.after(60000, update_all)
def add_event(new_title, new_target_date):
"""새 일정 추가"""
new_event = {
"id": str(uuid.uuid4()),
"title": new_title,
"target_date": new_target_date,
}
event_data.append(new_event)
save_events()
update_all()
def update_event(event_id, new_title, new_target_date):
"""기존 일정 수정"""
for event in event_data:
if event["id"] == event_id:
event["title"] = new_title
event["target_date"] = new_target_date
break
save_events()
update_all()
def delete_event(event_id):
"""일정 삭제"""
global event_data
event_data = [ev for ev in event_data if ev["id"] != event_id]
save_events()
update_all()
# Tkinter 메인 윈도우 생성
root = tk.Tk()
root.overrideredirect(True) # 기본 창 테두리 제거
root.attributes("-alpha", 0.7) # 투명도
root.configure(bg="gray")
root.attributes("-transparentcolor", "gray")
container_frame = tk.Frame(root, bg="gray")
container_frame.pack(padx=10, pady=10, anchor="center")
add_to_startup()
# JSON 파일 로드
event_data = load_events()
event_frames = []
# 최초 UI 초기화
update_all()
# 창 위치 지정
root.update_idletasks()
win_width = container_frame.winfo_reqwidth() + 20
win_height = container_frame.winfo_reqheight() + 20
screen_width = root.winfo_screenwidth()
x_pos = screen_width - win_width - 10
y_pos = 50
root.geometry(f"{win_width}x{win_height}+{x_pos}+{y_pos}")
root.mainloop()