-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathcore.py
More file actions
404 lines (344 loc) · 15.2 KB
/
core.py
File metadata and controls
404 lines (344 loc) · 15.2 KB
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
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
from PyQt5.QtCore import QTimer, QElapsedTimer, QObject, pyqtSignal
from utils import CrashHandler, Logger, _DATA_DIR
_logger = Logger(__name__)
class PairTimer(QObject):
"""实现一个两个计时器的类,一个计时器工作,一个计时器休息"""
# 定义信号
work_timer_timeout = pyqtSignal(int) # 发送工作时间(秒)
relax_timer_timeout = pyqtSignal(int) # 发送休息时间(秒)
def __init__(self):
super().__init__()
self.work_timer = QTimer() # 用于触发计时更新
self.relax_timer = QTimer()
self.work_elapsed = QElapsedTimer() # 用于记录实际经过的时间
self.relax_elapsed = QElapsedTimer()
self._work_time = 0 # 累计工作时间(秒)
self._relax_time = 0 # 累计休息时间(秒)
self._start_time = None # 开始时间
# 设置定时器间隔并连接信号
self.work_timer.setInterval(100) # 100毫秒刷新一次
self.relax_timer.setInterval(100) # 100毫秒刷新一次
self.work_timer.timeout.connect(self._on_work_timer)
self.relax_timer.timeout.connect(self._on_relax_timer)
def _on_work_timer(self):
"""工作定时器触发时发送信号"""
seconds = self.get_work_time() # 已经是秒
self.work_timer_timeout.emit(seconds)
def _on_relax_timer(self):
"""休息定时器触发时发送信号"""
seconds = self.get_relax_time() # 已经是秒
self.relax_timer_timeout.emit(seconds)
def start_work(self):
"""开始工作计时"""
if not self.work_elapsed.isValid():
self.work_elapsed.start()
self.work_timer.start()
self._start_time = datetime.now()
self.relax_timer.stop()
if self.relax_elapsed.isValid():
self._relax_time += self.relax_elapsed.elapsed() // 1000 # 转换为秒
self.relax_elapsed.invalidate()
def start_relax(self):
"""开始休息计时"""
if not self.relax_elapsed.isValid():
self.relax_elapsed.start()
self.work_timer.stop()
self._start_time = datetime.now()
self.relax_timer.start()
if self.work_elapsed.isValid():
self._work_time += self.work_elapsed.elapsed() // 1000 # 转换为秒
self.work_elapsed.invalidate()
def stop(self):
"""停止所有计时"""
self._start_time = None
self.work_timer.stop()
self.relax_timer.stop()
if self.work_elapsed.isValid():
self._work_time += self.work_elapsed.elapsed() // 1000 # 转换为秒
self.work_elapsed.invalidate()
if self.relax_elapsed.isValid():
self._relax_time += self.relax_elapsed.elapsed() // 1000 # 转换为秒
self.relax_elapsed.invalidate()
def get_work_time(self):
"""获取工作时间(秒)"""
current = self.work_elapsed.elapsed() // 1000 if self.work_elapsed.isValid() else 0 # 转换为秒
return self._work_time + current
def get_relax_time(self):
"""获取休息时间(秒)"""
current = self.relax_elapsed.elapsed() // 1000 if self.relax_elapsed.isValid() else 0 # 转换为秒
return self._relax_time + current
def get_start_time(self):
"""获取开始时间"""
return self._start_time
def reset(self):
"""重置所有计时器"""
self.work_timer.stop()
self.relax_timer.stop()
self.work_elapsed.invalidate()
self.relax_elapsed.invalidate()
self._work_time = 0
self._relax_time = 0
def clear(self):
"""清空所有计时器"""
self._work_time = 0
self._relax_time = 0
if self.work_elapsed.isValid():
self.work_elapsed.restart()
if self.relax_elapsed.isValid():
self.relax_elapsed.restart()
def get_total_time(self):
"""获取总时间(秒)"""
return self.get_work_time() + self.get_relax_time()
def get_elapsed_time(self):
"""获取当前计时器的时间(秒)"""
if self.work_elapsed.isValid():
return self.work_elapsed.elapsed() // 1000 # 转换为秒
elif self.relax_elapsed.isValid():
return self.relax_elapsed.elapsed() // 1000 # 转换为秒
return 0
def add_work_time(self, seconds: int):
"""
手动添加工作时间
Args:
seconds: 要添加的时间(秒)
"""
# 如果当前正在计时,先保存当前计时
if self.work_elapsed.isValid():
self._work_time += self.work_elapsed.elapsed() // 1000 # 转换为秒
self.work_elapsed.restart()
self._work_time += seconds
self._work_time = self._work_time if self._work_time > 0 else 0
# 触发一次信号更新显示
self._on_work_timer()
def add_relax_time(self, seconds: int):
"""
手动添加休息时间
Args:
seconds: 要添加的时间(秒)
"""
# 如果当前正在计时,先保存当前计时
if self.relax_elapsed.isValid():
self._relax_time += self.relax_elapsed.elapsed() // 1000 # 转换为秒
self.relax_elapsed.restart()
self._relax_time += seconds
self._relax_time = self._relax_time if self._relax_time > 0 else 0 # 防止出现负数
# 触发一次信号更新显示
self._on_relax_timer()
def is_work_active(self) -> bool:
"""
检查工作计时器是否在运行
Returns:
bool: 是否在运行
"""
return self.work_timer.isActive()
def is_relax_active(self) -> bool:
"""
检查休息计时器是否在运行
Returns:
bool: 是否在运行
"""
return self.relax_timer.isActive()
def is_any_active(self) -> bool:
"""
检查是否有任何计时器在运行
Returns:
bool: 是否有计时器在运行
"""
return self.is_work_active() or self.is_relax_active()
import json
from datetime import datetime, date, timedelta
import os
from typing import Dict, List, Tuple, Optional
from pathlib import Path
from utils import get_work_date
class DataManager:
"""管理每日卷摆时间数据"""
def __init__(self, is_file_operations_disabled: bool = False):
"""
初始化数据管理器
Args:
is_file_operations_disabled: 是否禁用文件操作
"""
self.data_dir = str(_DATA_DIR) # 将 _DATA_DIR 转换为字符串
self._is_file_operations_disabled = is_file_operations_disabled
self._current_date = None # 记录当前正在操作的日期
self._current_date_data = None # 当前日期的数据
if not self._is_file_operations_disabled and not os.path.exists(self.data_dir):
os.makedirs(self.data_dir)
_logger.info(f"创建数据目录: {self.data_dir}")
# 初始化时切换到当前工作日期
self.change_work_date(get_work_date())
def _get_date_filename(self, target_date: date) -> str:
"""获取指定日期的数据文件名"""
return os.path.join(self.data_dir, f"{target_date.strftime('%Y-%m')}.json")
def _load_month_data(self, target_date: date) -> Dict:
"""加载指定月份的数据"""
filename = self._get_date_filename(target_date)
if os.path.exists(filename):
with open(filename, 'r', encoding='utf-8') as f:
return json.load(f)
return {}
def change_work_date(self, target_date: date) -> bool:
"""
切换工作日期,如果日期发生变化则重新加载数据
Args:
target_date: 目标日期
Returns:
bool: 日期是否发生变化
"""
if self._current_date == target_date:
return False
_logger.info(f"切换到新的工作日期: {target_date}")
self._current_date = target_date
# 加载新日期的数据
month_data = self._load_month_data(target_date)
date_str = target_date.strftime('%Y-%m-%d')
if date_str not in month_data:
_logger.info(f"创建新的日期数据: {date_str}")
month_data[date_str] = {
'work_time': 0,
'relax_time': 0,
'segments': []
}
# 保存新创建的数据
if not self._is_file_operations_disabled:
self._save_month_data(target_date, month_data)
self._current_date_data = month_data[date_str]
return True
def _save_month_data(self, target_date: date, data_month: Dict):
"""保存月度数据"""
if self._is_file_operations_disabled:
_logger.debug("文件操作已禁用,跳过数据保存")
return
filename = self._get_date_filename(target_date)
with open(filename, 'w', encoding='utf-8') as f:
json.dump(data_month, f, ensure_ascii=False, indent=2)
_logger.debug(f"保存月度数据到文件: {filename}")
def save_current_state(self, work_time: int, relax_time: int):
"""
保存当前状态
Args:
work_time: 当前工作时间(秒)
relax_time: 当前休息时间(秒)
"""
if self._is_file_operations_disabled:
_logger.debug("文件操作已禁用,跳过状态保存")
return
# 确保在正确的日期下操作
self.change_work_date(get_work_date())
# 更新今日数据
self._current_date_data['work_time'] = work_time
self._current_date_data['relax_time'] = relax_time
# 更新月度数据文件
month_data = self._load_month_data(self._current_date)
date_str = self._current_date.strftime('%Y-%m-%d')
month_data[date_str] = self._current_date_data
self._save_month_data(self._current_date, month_data)
_logger.debug(f"保存当前状态 - 日期: {date_str}, 工作时间: {work_time}秒, 休息时间: {relax_time}秒")
def add_time_segment(self, start_time: datetime, end_time: datetime,
work_time: int, relax_time: int, is_work: bool, elapsed_time: int):
"""
添加一个时间段的数据
Args:
start_time: 开始时间
end_time: 结束时间
work_time: 工作时间(秒)
relax_time: 休息时间(秒)
is_work: 是否是工作时间段
elapsed_time: 当前计时器的增量时间(秒)
"""
if self._is_file_operations_disabled:
_logger.debug("文件操作已禁用,跳过添加时间段")
return
# 使用结束时间来决定数据属于哪一天
target_date = get_work_date(end_time)
# 切换到目标日期
self.change_work_date(target_date)
segment = {
'start': start_time.strftime('%Y-%m-%d %H:%M:%S'),
'end': end_time.strftime('%Y-%m-%d %H:%M:%S'),
'work_time': work_time,
'relax_time': relax_time,
'type': 'work' if is_work else 'relax',
'elapsed_time': elapsed_time
}
self._current_date_data['segments'].append(segment)
self._current_date_data['work_time'] = work_time
self._current_date_data['relax_time'] = relax_time
# 保存数据
month_data = self._load_month_data(self._current_date)
date_str = self._current_date.strftime('%Y-%m-%d')
month_data[date_str] = self._current_date_data
self._save_month_data(self._current_date, month_data)
_logger.info(
f"添加时间段 - 日期: {date_str}, "
f"类型: {'工作' if is_work else '休息'}, "
f"开始: {segment['start']}, "
f"结束: {segment['end']}, "
f"工作时间: {work_time}秒, "
f"休息时间: {relax_time}秒, "
f"实际经过时间: {segment['elapsed_time']}秒"
)
def get_day_stats(self, target_date: Optional[date] = None) -> Dict:
"""
获取指定日期的统计数据
Args:
target_date: 目标日期,默认为今天
Returns:
包含工作和休息时间的字典
"""
if target_date is None:
target_date = get_work_date()
# 如果是查询当前日期的数据,直接返回
if target_date == self._current_date:
return {
'work_time': self._current_date_data['work_time'],
'relax_time': self._current_date_data['relax_time'],
'segments': self._current_date_data.get('segments', [])
}
# 否则从文件加载数据
month_data = self._load_month_data(target_date)
date_str = target_date.strftime('%Y-%m-%d')
if date_str in month_data:
return {
'work_time': month_data[date_str]['work_time'],
'relax_time': month_data[date_str]['relax_time'],
'segments': month_data[date_str].get('segments', [])
}
return {'work_time': 0, 'relax_time': 0, 'segments': []}
def get_month_stats(self, year: int, month: int) -> List[Dict]:
"""
获取指定月份的统计数据
Args:
year: 年份
month: 月份
Returns:
月度统计数据列表
"""
target_date = date(year, month, 1)
month_data = self._load_month_data(target_date)
stats = []
for date_str, data in month_data.items():
stats.append({
'date': date_str,
'work_time': data['work_time'],
'relax_time': data['relax_time'],
'segments': data.get('segments', [])
})
return sorted(stats, key=lambda x: x['date'])
def get_recent_days_stats(self, days: int = 7) -> List[Dict]:
"""
获取最近几天的统计数据
Args:
days: 天数
Returns:
最近几天的统计数据列表
"""
today = get_work_date()
stats = []
for i in range(days):
target_date = today.fromordinal(today.toordinal() - i)
day_stats = self.get_day_stats(target_date)
day_stats['date'] = target_date.strftime('%Y-%m-%d')
stats.append(day_stats)
return list(reversed(stats)) # 按日期升序返回