Some checks failed
CI / test (push) Has been cancelled
핵심 기능: - 단축근무·표준·반일 등 다양한 근무 패턴 (5개 프리셋 + 사용자 정의) - Windows 이벤트 뷰어 자동 출퇴근 감지 - 30분 단위 연장근무 적립/사용 시스템 - 1.0/0.5/0.25일 연차·반차·반반차 - 자동 점심·저녁·외출·자동 백업·화면 잠금 자동 외출 - 한국 공휴일 자동 등록 (음력 포함, holidays 패키지) - matplotlib 차트 기반 주간/월간/패턴 통계 - 미니 위젯 + 시스템 트레이 통합 - 한국어/English i18n - 자가 업데이트 (updater.exe + Gitea Releases) 아키텍처: - core/ (db, time_calculator, notifier, i18n, version, settings_keys) - ui/ (main_window + 9 dialogs + 3 controllers) - utils/ (backup, lock_detector, debug_log, updater_client, time_format) - tests/ (66 pytest 단위) + 통합/i18n GUI 검증 CI/CD: - .gitea/workflows/ci.yml: push 시 pytest + 통합 테스트 - .gitea/workflows/release.yml: v* 태그 push 시 두 .exe 자동 빌드 + Releases 첨부 Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
56 lines
1.7 KiB
Python
56 lines
1.7 KiB
Python
"""
|
|
자동 점심시간 적용 컨트롤러.
|
|
|
|
조건: auto_lunch=true, 출근 후 4시간 이상, 점심 미적용, 오늘 미적용.
|
|
주말/공휴일은 스킵. 1Hz hot path에서 호출되므로 캐시 사용.
|
|
"""
|
|
from __future__ import annotations
|
|
from datetime import datetime
|
|
|
|
from core.settings_keys import AUTO_LUNCH
|
|
|
|
|
|
class AutoLunchManager:
|
|
"""update_display() 1Hz tick에서 호출."""
|
|
|
|
def __init__(self, window):
|
|
self.window = window
|
|
self.db = window.db
|
|
self._enabled_cache = None # None=미로딩, True/False=캐시값
|
|
self._non_working_cache = None
|
|
self._non_working_date = None
|
|
|
|
def invalidate(self) -> None:
|
|
"""설정 변경 시 캐시 무효화 (reload_settings에서 호출)."""
|
|
self._enabled_cache = None
|
|
|
|
def maybe_apply(self, now: datetime) -> None:
|
|
w = self.window
|
|
if w.auto_lunch_applied_today or w.lunch_break_enabled:
|
|
return
|
|
|
|
if self._enabled_cache is None:
|
|
self._enabled_cache = (
|
|
self.db.get_setting(AUTO_LUNCH, 'false').lower() == 'true'
|
|
)
|
|
if not self._enabled_cache:
|
|
return
|
|
|
|
today = now.date()
|
|
if self._non_working_date != today:
|
|
self._non_working_cache = w.time_calc.is_non_working_day(now, self.db)
|
|
self._non_working_date = today
|
|
if self._non_working_cache:
|
|
return
|
|
|
|
elapsed = now - w.clock_in_time
|
|
if elapsed.total_seconds() < 4 * 3600:
|
|
return
|
|
|
|
w.auto_lunch_applied_today = True
|
|
w.lunch_break_enabled = True
|
|
w.lunch_button.setChecked(True)
|
|
w.update_lunch_status()
|
|
if w.is_clocked_in:
|
|
self.db.update_lunch_break(today.isoformat(), True)
|