KINDNICK f5751460e3 v2.11.2: 통계 차트 frozen 실제 수정(numpy _multiarray_tests) + 도전과제 라이트 가독성
- fix: frozen main.exe에서 numpy.core._multiarray_tests 누락으로 matplotlib import 실패 → 차트 폴백. spec에 hiddenimport 추가 + numpy.testing 제외 제거 (디버그 로그로 원인 확인)
- fix: 도전과제 라이트 테마 헤더 숫자/배지/진행 텍스트/바 대비 개선

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
2026-06-04 19:22:00 +09:00

28 KiB
Raw Blame History

Changelog

All notable changes to this project will be documented in this file.

The format is based on Keep a Changelog.

[2.11.2] — 2026-06-04

Fixed

  • 통계 차트가 빌드(main.exe)에서 안 뜨던 진짜 원인 — frozen 빌드에서 numpy C-확장 numpy.core._multiarray_tests가 누락(numpy.testing 제외의 영향)되어 matplotlib import가 ModuleNotFoundError로 실패 → "matplotlib 필요" 폴백. main.spec에 해당 모듈 hiddenimport 추가 + numpy.testing 제외 제거. (디버그 로그로 원인 확인: chart_widget이 실패 사유를 기록)
  • 도전과제 라이트 테마 가독성 — 헤더 강조 숫자/등급 배지/진행 숫자/진행 바를 라이트에서 대비 높은 색으로 조정 (다크는 기존 비비드 색 유지).

[2.11.1] — 2026-06-04

Fixed

  • 빌드(main.exe)에서 통계 차트가 표시되지 않던 문제 — frozen 빌드는 PyInstaller가 matplotlib QtAgg(backend_qtagg)만 번들하는데 chart_widgetbackend_qt5agg를 import해 실패 → "matplotlib 필요" 폴백만 보였음. backend_qtagg 우선 import(+ qt5agg 폴백) + 실패 원인 로깅, main.specbackend_qtagg/PyQt5.sip 명시.
  • 통계·도움말·도전과제 화면이 라이트 테마에서도 다크로 고정되던 문제dark_components와 세 화면(+통계 차트 배경/그리드/텍스트)을 현재 테마(ThemeColors)에 따르도록 변경. 다크 기본값은 그대로, 라이트 전환 시 함께 라이트로. 다크 등급 카드/차트 막대 등 강조색은 유지.

[2.11.0] — 2026-06-04

Changed — UI 전면 다크 리디자인

  • 모던 다크 미니멀 테마(Notion/Linear 톤): 배경 #1A1B1E / 카드 #25262B / 보더 #2C2E33, 단일 포인트 컬러 #4DABF7(주요 버튼·포커스 전용), 텍스트 #E9ECEF/#909296
  • 다크가 기본 테마 (신규 설치 기준; 기존 사용자가 고른 설정은 보존)
  • 번들 폰트 NanumSquare (font/, utils/font_loader.py) — OS 미설치 시 Malgun Gothic 폴백, main.spec에 동봉
  • 통일 여백(외곽 24 / 위젯 12 / 카드 16), border-radius 8px, 버튼 그라데이션·베벨 제거(flat), 입력 포커스 시 보더 컬러만 accent, 진행률 바 6px
  • 남은시간 히어로 영역(출근/현재 한 줄 + 예상 퇴근시각 통합), 퇴근 가능 시 그린(#51CF66) 피드백

Added

  • 라인 아이콘 시스템 (ui/icons.py, QtSvg) — 이모지 대신 테마 틴팅 모노크롬 라인 아이콘. 하단 네비 / 통계 카드 / 트레이·미니위젯 메뉴 등 전반 적용 (main.specPyQt5.QtSvg 포함)
  • 연장근무 적립 기록 삭제 — 연장근무 관리의 적립 내역 우클릭 → 삭제 (Database.delete_overtime_earned)

Fixed

  • 자동 적립(auto_overtime) OFF가 자동 퇴근 경로에서 무시되던 버그 — 근무일 경계 롤오버 / 이전일 자동 퇴근 처리도 설정을 존중하도록 게이팅 (_apply_auto_overtime_gate). (clock_out 대화상자 '아니오' 경로는 정상이었음)
  • 다크 테마 깨짐: 테이블 세로 헤더·코너 버튼 흰색 누수, 도움말 탭 상단 흰 라인(documentMode), 트레이/미니위젯 우클릭 메뉴 미적용(검정 글씨) 수정
  • 앱 전반 UI 크롬 이모지 제거 + 색상 팔레트 정합 (일일보고/Discord 텍스트는 유지)

Tests

  • tests/test_overtime_accrual_guard.py 추가 — 적립 가드 2건(OFF=미적립 / ON=적립) + 적립 삭제 1건

[2.10.2] — 2026-05-16

Fixed

  • 휴일/주말 근무 시 카운터 초가 항상 00 으로 멈춰 보이던 문제 (사용자 보고)
    • 원인: 휴일 분기에서 calculate_holiday_overtime의 분 절삭값(적립 단위)을 그대로 표시에 사용 → 초 정보 소실
    • 수정: 표시용 remaining을 초 정밀도 timedelta로 분리 계산 (적립 계산은 퇴근 시 분 단위 그대로 — 영향 없음)
    • 차감 항목(점심·저녁·외출·연장 사용)은 calculate_holiday_overtime과 동일하게 적용

[2.10.1] — 2026-05-01

Fixed — 업데이트 시 cmd 창 깜빡임 제거

  • updater.spec: console=Trueconsole=False (windowed 빌드). 자동 업데이트 적용 시 잠깐 뜨던 까만 cmd 창이 더 이상 보이지 않음.
  • updater.py: stderr 출력을 ~/.clockout_logs/updater.log 파일 폴백으로 전환 — windowed 모드라도 진단 로그는 보존. 모든 단계(시작/PID 대기/replace/launch) 에 타임스탬프 + 결과 기록.
  • updater.py launch(): subprocess.PopenCREATE_NO_WINDOW 플래그 추가 (DETACHED_PROCESS와 함께) — 자식 프로세스가 콘솔을 새로 만들지 않음.
  • utils/updater_client.py apply_update(): 같은 패턴으로 CREATE_NO_WINDOW 추가. main.exe → updater.exe 호출 시점에서도 콘솔 생성 차단.

[2.10.0] — 2026-05-01

Added — 정부 공휴일 API 자동 동기화

  • 공공데이터포털 특일정보 API 연동 (utils/holiday_api.py)
    • 한국천문연구원 운영 공식 데이터 — /getRestDeInfo 엔드포인트
    • 임시공휴일·근로자의 날까지 정부 공인 데이터로 보강
    • 일일 한도 10,000회 / 사용자 50명 = 0.5% 사용
    • 키는 dev 본인 계정의 특일정보 API 한정 키
  • Database.add_korean_holidays_from_api(year) — 정부 API 1차 시도
  • add_korean_holidays_auto() 동작 변경 — 1차 정부 API → 2차 fallback holidays 패키지
  • migrate_v290_holidays_auto_sync — 일 1회 자동 동기화 (백그라운드 스레드)
    • sentinel: settings['holidays_synced_date']
    • 매일 호출 → 정부가 임시공휴일 발표하면 다음 날 자동 반영
    • 부트스트랩 비차단 (네트워크 호출은 daemon thread)
    • 테스트 환경: CLOCKOUT_DISABLE_HOLIDAY_SYNC=1 로 비활성화

Changed

  • 설정 → "한국 공휴일 자동 추가" 버튼 안내문 — 1차 정부 API / 2차 holidays 패키지

Tests

  • tests/test_holiday_api.py 14개 신규 (응답 파싱 / 단일/다중 item / 401·timeout / 응답 검증)
  • tests/conftest.py — 모든 테스트에서 백그라운드 동기화 비활성화
  • pytest: 175 → 189

주의

  • 키 활용기간 시작 직후엔 백엔드 propagation으로 401 가능 (1~2시간 또는 익일 활성화). 401 시 fallback (holidays 패키지 + 근로자의 날 명시 추가) 정상 동작 — 사용자 영향 없음.

[2.9.0] — 2026-05-01

Fixed — 휴일 hot-path 버그 (사용자 보고)

  • 휴일에 출근해도 정상 출근으로 처리되어 추가근무 적립이 안 되던 문제
    • update_display() 1Hz 루프에 is_non_working_day 분기 누락으로 휴일에도 "남은 시간 8h"부터 카운트다운 → 실제 출근 즉시 적립이 시작되지 않음
    • 수정: 출근 직후부터 음수 remaining 표시, "공휴일 근무 (전체 적립)" 그룹 타이틀
    • 진행바: 휴일은 100% 고정 (의미 없음)
    • 예상 퇴근: "휴일 근무 (정해진 퇴근시각 없음)"
  • 휴일 "퇴근 30분 전" 알림 게이팅 — 휴일엔 정해진 퇴근시각이 없으니 무의미한 알림 스킵
  • 자동복구 퇴근 3곳의 // 30) * 30 하드코딩 → 사용자 overtime_unit (15/30/60) 설정 적용
  • 4곳에 중복되던 휴일 연장 계산 로직을 TimeCalculator.calculate_holiday_overtime() 헬퍼로 통합

Added — 연차 미리등록 + 통합 스케줄 + 반복 연차 (Phase 1+2)

Phase 1 — 연차 미리등록 + 자동 적용

  • DB: get_leave_minutes_for(date) / has_full_day_leave(date) / get_leave_records_by_date(date) / get_leave_records_by_range(start, end)
  • TimeCalculator: effective_work_minutes(date_obj, db) — 부분 연차만큼 정규 근무 차감
  • 종일 연차일 자동 처리:
    • 자동 출근감지 스킵 (event_monitor 호출 안 함)
    • "🌴 오늘은 휴가" 카드 표시, 카운트다운 제거
    • 메인/미니 위젯/트레이 모두 일관된 휴가 상태 표시
  • 종일 연차 + 출근 override: 휴일처럼 전체 시간 적립 (사용자 확인 후)
  • 부분 연차 (반차/반반차/시간): 기존 leave_used 경로로 카운트다운 단축
  • AddLeaveDialog 검증 강화: 미래 1년 setMaximumDate / 주말·공휴일 차단 / 같은 날 1일 초과 차단
  • leave_calendar_view: 예정(파랑) / 사용완료(녹·노·보) 색상 분리

Phase 2 — 통합 스케줄 + 반복 연차

  • recurring_leaves 테이블 (pattern/leave_type/days/start_date/end_date/memo)
  • core/recurring_leaves.py: weekly / biweekly / monthly 패턴 파서 + expand_for_range/date
  • 자동 합산: get_leave_minutes_for() / has_full_day_leave()가 반복 패턴 인스턴스도 함께 검사
  • ui/recurring_leave_dialog.py: 매주/격주 요일 또는 매월 N일 입력
  • ui/schedule_view.py: 월간 통합 캘린더 (휴일·연차·반복 색상 구분 + 우클릭 삭제)
  • 진입점: MainWindow.show_schedule(), 트레이 "🗓️ 스케줄", LeaveView "🗓️ 스케줄"

Changed

  • 근로자의 날(5/1) 자동 추가holidays.KR 패키지가 누락하는 노동자 휴일을 add_korean_holidays_auto()에서 명시적 보강 (매년 반복)

Tests

  • pytest: 122 → 175 (+53)
    • tests/test_recurring_leaves.py 32개 (패턴 파싱/매칭/expand/describe)
    • tests/test_database.py +12 (TestLeaveQueriesByDate + TestRecurringLeavesDB)
    • tests/test_time_calculator.py +9 (TestHolidayOvertime)
  • 통합 시나리오: 48 → 53 (+5)
    • S52A 휴일 hot-path / S52B 종일 연차 / S52C 반복 패턴 / S52D 반차 effective / S52E 종일 effective

[2.8.0] — 2026-05-01

Added — 도전과제 시스템 + 디자인 리뉴얼

  • 🏆 도전과제 시스템 (153개 자동 평가) — 출근·퇴근·연장·연차·식사·외출·계절·시간대·시크릿 등 16개 카테고리.
    • core/achievements.py: Achievement dataclass + evaluate_all(db) + sync_definitions_to_db(db).
    • 5분 throttle로 자동 평가, 신규 잠금 해제 시 시스템 알림 + Discord embed push.
    • achievements 테이블 확장: code(UNIQUE), category, tier, is_secret, progress, target, created_at 컬럼 추가 (idempotent 마이그레이션).
    • 5단계 등급: 🥉 브론즈 / 🥈 실버 / 🥇 골드 / 💎 플래티넘 / 🌟 레전드.
    • 시크릿 9개 (회문, 잭팟 시각, 13일의 금요일, 7-7-7, 정확 8시간, π day, 피보나치 등).
    • 메타 도전과제 (도전과제 자체 달성 카운트).
  • ui/achievements_view.py — 4탭 다이얼로그 (전체 / 진행 중 / 완료 / 시크릿).
    • 등급별 그라디언트 카드, 진행 게이지, 카테고리 태그, 시크릿 처리.
  • 자동 hire_date 추적 — 첫 add_work_record 호출 시 settings에 자동 기록 (1주년, 365일 후 출근 등 도전과제 활성화).
  • 뷰 진입 카운터stat_*_view_count, calendar_view_count, daily_report_count 등 8개 settings 키. 도전과제 + 사용 통계용.

Changed — 다크 테마 디자인 리뉴얼

  • ui/dark_components.py 신설 — 재사용 가능한 다크 디자인 컴포넌트:
    • dialog_qss(), tabs_qss(), scroll_qss(), button_qss(variant).
    • build_gradient_header(), build_stat_card(), build_section_card().
    • style_progressbar(), transparent_label() — 글로벌 QSS 충돌 회피.
    • 카드 테마 7종: blue / cyan / green / gold / pink / red / gray.
  • ui/stats_view.py — 4분할 카드 + 차트 섹션 카드. 골드 강조 탭. ghost 닫기 버튼.
  • ui/help_view.py — 다크 톤 + HelpHTML 내부에 다크 CSS 주입 (h1/h2/h3, code, blockquote, table 컬러).
  • ui/chart_widget.py — 모든 matplotlib 차트 다크 테마 적용 (figure/axes facecolor, grid, ticks, legend).
  • 온보딩 위저드 — 저녁 분(minutes) 입력 옵션 + i18n화 (Korean/English 프리셋 라벨).

Fixed — 안정성·일관성

  • 타임존 자정 경계 버그: has_notification_todayCURRENT_TIMESTAMP(UTC) vs DATE('now', 'localtime') mismatch로 KST 0~9시 사이 알림 중복 발송 가능 — DATE(sent_at, 'localtime') 양쪽 적용.
  • DB 연결 누수 가드_conn() 컨텍스트 매니저 도입, 40+ 메서드 변환 (직접 get_connection() 호출 0건).
  • DB 이중 부트스트랩 제거MainWindow(db=None) 옵션 인자 추가, main.py가 만든 db를 재사용.
  • crash_handler 폴백 — DB 로깅/다이얼로그 단계 분리, 모두 실패해도 ~/.clockout_logs/crashes.log에 기록.
  • updater PID race window — 지수 backoff 재시도 (0.3→4.8s, 총 ~9초). Windows Defender 락 해제 대기 충분.
  • Discord URL 형식 검증 — Snowflake ID 17~20자리 + 50+자 토큰 정규식.
  • MealTimeDialog — 출근 시각 범위 검증 + 야간 출근자 자정 경계 자동 처리.
  • CSV importer/exporterdinner_minutes 컬럼 round-trip 지원.
  • 일일 보고서 — 저녁 섹션 추가 (이전엔 점심만 표시), break_type 분리, 자정 경계 처리.
  • 저녁 알림check_dinner_reminder() 신규 (점심 알림과 대칭).
  • 알림 임계값 설정화 — 5개 하드코딩 값(점심/저녁 알림 시간, 연장 누적, 주간 한도, 연속 야근)을 settings로 노출.
  • closeEvent 정리aboutToQuit 시그널로 timer/notifier/tray 정리.
  • 마이그레이션 일관성 — 12개 마이그레이션 모두 _conn() + try/finally 보장.

DB 인덱스 (성능)

  • idx_break_records_date_type, idx_break_records_date.
  • idx_overtime_bank_date, idx_overtime_usage_date, idx_leave_records_date.

Tests

  • pytest 116개 PASS, 통합 시나리오 44/48 PASS (PyQt 환경 4개 제외).
  • 도전과제 한 사이클 시나리오 검증 (30일 시뮬레이션 → 19개 자동 잠금 해제).

[2.7.0] — 2026-04-30

Added — 폴리싱 릴리스 (사용자 가시 변화는 작지만 i18n + 테스트 + 구조 개선)

  • i18n 사전 100% 커버리지break_view, overtime_view, leave_view, clock_in_dialog 의 내부 라벨/메시지/플레이스홀더까지 전부 ko/en 키화. 새 키 50+개 추가 (view.break.*, view.overtime.*, view.leave.*, dlg.*).
  • i18n 런타임 재번역 — 재시작 없이 메인 화면 즉시 언어 전환.
    • ui/i18n_runtime.py: register(widget, key) + set_language_and_retranslate(lang)
    • 위젯은 weakref로 보관되어 삭제 시 자동 정리
    • 메인 윈도우 타이틀/하단 메뉴 5개 버튼 등록 완료 (점진 확대)
    • 일부 다이얼로그는 여전히 재시작 필요 (다음 릴리스에서 점진 등록)
  • MealController 분리main_window.py 에서 점심/저녁 토글·라벨 갱신 로직을 ui/controllers/meal_controller.py 로 추출. 기존 LockMonitor/AutoLunch/ NotificationOrchestrator 패턴 준수.

Tests

  • 통합 테스트 +15 시나리오 (S36S52): 온보딩 신규/기존, salary 추정, CSV 가져오기 (skip/overwrite/overtime 적립까지), notification_log dedupe, meal_record, crash_log, updater semver 비교, Discord 입력 검증, goal/accessibility 키 등.
  • 신규 pytest 모듈 4종 — test_salary.py, test_csv_importer.py, test_discord_webhook.py(network mocked), test_crash_handler.py.
  • test_i18n_runtime.py — register/retranslate/post-callback/dead-widget 정리 검증.
  • 총 pytest 케이스: 90 → 122, 통합 시나리오: 33 → 48.

Docs

  • README.md v2.4v2.6 신기능 섹션 정리, 재번호.
  • CLAUDE.md v2.6+ 아키텍처 — 컨트롤러 모듈, 35+ 설정 키, 릴리스 플로우 반영.
  • INSTALL.md 최신 단일파일 배포 + 온보딩 + 디스코드 + 환경변수 정리.
  • AGENTS.md 10+ DB 테이블, 컨트롤러 맵, Past Incidents 갱신.

[2.6.0] — 2026-04-30

Added — Phase 4 (3종)

  • 글꼴 크기 조절 (100% / 125% / 150%) — 설정에서 즉시 반영
  • 고대비 모드 — 검정 배경 + 노란 텍스트 (시각약자/야간)
  • 코드 서명 인프라release.ps1에 Authenticode 서명 단계 추가 (옵션)
    • $env:CODE_SIGN_CERT (.pfx 경로) + $env:CODE_SIGN_PASS 환경변수 설정 시 자동 서명
    • signtool.exe 없거나 cert 미설정 시 자동 스킵
    • 코드 서명 인증서 확보 후 활성화하면 SmartScreen 경고 제거 가능

Settings (신규)

  • font_scale, high_contrast

[2.5.0] — 2026-04-30

Added — Phase 3 (4종)

  • 주간 자동 리포트 — 월요일 첫 출근 시 (또는 첫 5분 tick) 지난주 요약 발송
    • 시스템 알림 + Discord push (옵션) 동시
    • notification_log로 중복 발송 방지
    • 항목: 총 근무·일평균·연장근무·가장 긴 날
  • matplotlib 차트 호버 디테일 — 막대 위에 마우스 올리면 정확한 수치 툴팁
    • 일별 근무 시간 차트(주간 탭)에 적용
  • 출근 시각 분포 차트 — 패턴 분석 탭에 30분 단위 히스토그램
    • 평균 출근 시각 빨간 점선으로 표시
  • 휴가 캘린더 시각화 — 연차 관리 → "📅 캘린더 보기"
    • 사용 일자에 종일/반차/반반차별 색상 표시
    • 날짜 클릭 → 사용 내역 표시

Fixed

  • leave_view.py setLayout 들여쓰기 회귀 수정

[2.4.0] — 2026-04-30

Added — Phase 2 (5종)

  • 점심/저녁 실제 시간 입력 — 점심/저녁 버튼 우클릭 → 시작·종료 시각 입력 다이얼로그
    • 자동 60분 대신 정확한 분 단위 기록 (break_records.break_type='lunch'/'dinner')
  • 캘린더 우클릭 → 과거 일자 추가/편집/삭제
    • 비어있는 날짜 우클릭: "기록 추가" — 출/퇴근/점심/메모 입력
    • 기록 있는 날짜 우클릭: "편집"/"삭제"
  • 월간 목표 설정 + 진행률
    • 설정 → 월 연장근무 상한 (시간/분) + 일 평균 근무 목표 (시간)
    • 통계 → 월간 탭에 진행률 게이지 (60%/100% 임계 시 색상 변경)
    • 0=비활성 (비활성 시 위젯 자체 숨김)
  • CSV 가져오기 — 표준 포맷 date,clock_in,clock_out,lunch_minutes,memo
    • 충돌 정책: 덮어쓰기/건너뛰기/취소
  • 자동 Crash Report (Gitea Issues)
    • 전역 예외 후킹 → crash_log 저장 + 사용자에게 다이얼로그
    • "복사" / "Gitea에 보고" (PAT 옵션) — issue 자동 생성

Settings (신규 4개)

  • goal_overtime_max_monthly, goal_avg_hours_daily
  • gitea_feedback_token, gitea_feedback_enabled

[2.3.3] — 2026-04-30

Fixed

  • Discord 웹훅 항상 실패하던 문제 (Cloudflare error 1010 / 403 Forbidden)
    • 원인: Python 기본 User-Agent(Python-urllib/3.x)를 Discord/Cloudflare가 봇으로 인식해 차단
    • 수정: utils/discord_webhook.py에 브라우저 UA 헤더 추가 (Mozilla/5.0 (Windows NT 10.0; Win64; x64) ClockOutCalculator/2.3)
    • 이제 온보딩 위저드의 "테스트 메시지 보내기" + 출퇴근/휴식 push 모두 정상 동작

[2.3.2] — 2026-04-30

Fixed

  • 도움말 다이얼로그가 빈 창으로 표시되던 치명적 버그
    • v2.3.1에서 "온보딩 다시 보기" 버튼 추가 시 self.setLayout(main_layout) 라인이 실수로 _reopen_onboarding 메서드 안으로 들여쓰기되어 init_ui가 setLayout 없이 종료
    • F1 누르면 빈 화면만 나옴 (탭 6개 모두 미렌더링)
    • setLayout 호출 위치 복원 → 정상 표시

[2.3.1] — 2026-04-30

Fixed

  • 도움말 다이얼로그에 "온보딩 다시 보기" 버튼 추가
    • v2.3.0에서 show_onboarding() 메서드는 만들었지만 UI 진입점 누락
    • 이제 도움말 (F1) → 좌측 하단 "🚀 온보딩 다시 보기" 버튼으로 위저드 재실행 가능
    • 본인 검증/설정 변경 시 유용

[2.3.0] — 2026-04-30

Added — Phase 1 + E1 (소비자 친화 6종)

  • 첫 실행 온보딩 위저드 (강제) — 5단계: 환영 → 근무패턴 → 출근 감지 방식 → 연차/시급(옵션) → Discord(옵션) → 완료
    • 신규 사용자: 자동 표시 / 기존 사용자(work_records 있음): 자동 완료 처리
    • "도움말 → 온보딩 다시 보기" 메뉴로 언제든 재실행 가능
  • 시급 → 추정 급여 (옵션) — 포괄임금이 아닐 때만 활성화
    • 통계 화면 월간 탭에 "이번 달 추정 급여" 카드
    • 퇴근 후 "오늘 요약" 카드에도 추정 급여 표시
    • 연장수당 가산률 1.0 / 1.5 / 2.0 선택
  • 출퇴근 시각 인라인 편집 — 메인 화면 출근/퇴근 라벨 클릭 → 즉시 수정 다이얼로그
  • 퇴근 후 "오늘 요약" 카드 — 메인 화면 상단에 총 근무/점심/외출/연장/추정급여 표시
    • 다음 출근 시 자동 숨김 / X 버튼으로 수동 닫기
  • 장시간 근무 휴식 권고 알림 — 연속 N시간(기본 4시간) 자리 비움 없으면 "🌿 잠시 일어나세요" 토스트
    • 5분 throttle + 일 1회 가드 (notification_log 테이블)
  • Discord 웹훅 알림 (옵션) — 출퇴근/휴식권고 모바일 push
    • 봇 등록·서버 운영 0. 채널 웹훅 URL만 입력
    • 출근(녹색) / 퇴근 정시(파랑) / 퇴근 연장(주황) / 건강경고(분홍) embed
    • 온보딩에서 즉시 활성화 + "테스트 메시지" 버튼

Database

  • break_records.break_type 컬럼 추가 ('break' / 'lunch' / 'dinner' 구분)
  • notification_log 테이블 신규 (channel, event_type, sent_at, success — 중복 발송 가드 + 통계용)
  • 기존 사용자 onboarding_completed 자동 true 처리 마이그레이션

Settings (신규 11개)

  • onboarding_completed, salary_enabled, hourly_wage, overtime_rate
  • health_break_enabled, health_break_hours
  • discord_webhook_url, discord_notif_clock_in, discord_notif_clock_out, discord_notif_health

[2.2.4] — 2026-04-30

Added

  • auto_overtime 옵션 실제 동작 — 기존엔 UI만 있고 동작 안 했음
    • OFF: 퇴근 시 적립 가능한 분이 있으면 "적립할까요?" 다이얼로그 표시 (Y/N)
    • ON (기본): 자동 적립 (이전과 동일)
  • overtime_unit 실제 동작 — 30/15/60분 단위 적립 선택 반영
    • 기존엔 무조건 30분 절삭. 이제 설정값 사용 (time_calculator.calculate_overtime unit_minutes 파라미터)
  • 퇴근 알림 시점 사용자 설정 (notification_before_minutes)
    • 기존 30분 하드코드 → 1~120분 SpinBox로 사용자 지정
    • 설정 → 알림 그룹에 "퇴근 알림 시점" SpinBox

Removed (죽은 옵션 정리)

  • auto_detect_boot — 부팅 감지 비활성화 모드. 어디서도 안 쓰임
  • notification_enabled — 마스터 스위치. 4개 개별 NOTIF_* 키로 충분
  • annual_leave_usedleave_balance로 대체된 레거시 카운터

Fixed

  • 죽은 옵션이 사용자에게 "동작하는 것처럼" 보이던 신뢰성 문제 일괄 해소

[2.2.3] — 2026-04-30

Fixed

  • updater.exe 임베딩 빌드 안정화
    • v2.2.2에서 main.spec --clean이 빌드 시작 시 dist/updater.exe를 지우면서 임베딩이 누락되던 문제 수정
    • release.ps1dist/updater.exebuild/staging/로 복사 후 main.spec이 그걸 참조
    • main.exe 사이즈가 다시 ~78MB (updater 내장) 로 정상화

[2.2.2] — 2026-04-30

Added

  • 윈도우 제목에 현재 버전 표시 — 자동 업데이트 후 신/구 버전 시각적 구분
    • 예: ⏰ 퇴근시간 계산기 v2.2.2

Test

  • 자동 업데이트 더미 릴리스 — 단독 main.exe 배포 + 자가 업데이트 흐름 검증용

[2.2.1] — 2026-04-30

Added

  • updater.exe 내장 (Single-file 배포)
    • main.exe 안에 updater.exe를 PyInstaller datas로 임베드
    • 시작 시 main.exe가 같은 폴더에 updater.exe를 자동 추출 (없거나 사이즈 다르면 갱신)
    • 사용자는 이제 main.exe 하나만 받아도 자동 업데이트 동작
    • 업데이트로 main.exe가 교체되면 새 main 실행 시 새 updater.exe도 자동 갱신
  • utils/updater_client.py: _MEIPASS fallback — 권한 부족 등으로 추출 실패 시 TEMP에서 동작

Changed

  • release.ps1 빌드 순서: updater 먼저 → main 나중 (datas 의존성)
  • main.spec: dist/updater.exe 존재 시 자동 임베드 (조건부 datas)

Fixed

  • 한국어 릴리스 노트 인코딩 (Get-Content ANSI → [System.IO.File]::ReadAllText UTF-8)

[2.2.0] — 2026-04-30

Added

  • 자동 업데이트 — Gitea Releases 기반
    • 시작 5초 후 백그라운드 버전 체크 (silent)
    • F5 또는 설정 → 데이터 관리 → "업데이트 확인" 버튼으로 수동 트리거
    • 새 버전 발견 시 다운로드 → updater.exe가 메인 종료 대기 → 파일 교체 → 재시작
    • 실패 시 .bak 자동 롤백
  • core/version.py__version__ 상수
  • updater.py + updater.spec — 독립 자가 업데이터 (Python 표준 라이브러리만, 6MB)
  • utils/updater_client.py — Gitea/GitHub 호환 Releases API 클라이언트
  • release.ps1 — 로컬 원클릭 릴리스 스크립트 (Runner 불필요)
    • 빌드 + 태그 push + Gitea Release 생성 + 자산 업로드 자동화
    • CHANGELOG에서 릴리스 노트 자동 추출
    • --DryRun / --SkipTests 옵션

Changed

  • Settings → 데이터 관리에 "버전 표시 + 업데이트 확인" 추가
  • main.spec datas에 안내 주석 (updater.exe는 별도 배포)

[2.1.0] — 2026-04-29

Removed

  • Claude AI 분석 기능 제거 — 외부 API 의존성 정리
  • 로컬 HTTP API 제거 — 외부 위젯 연동 기능 미사용으로 정리
  • pandas 의존성 제거 — 코드에서 미사용 (PyInstaller 빌드 사이즈 감소)

Added

  • 첫 잠금 해제 = 출근 옵션 (clock_in_on_unlock) — PC를 안 끄는 사용자 케이스 해결
  • WAL 모드 + busy timeout — 클라우드 동기화 환경에서 다중 PC 동시 접근 안전성↑
  • MainWindow 컨트롤러 분리LockMonitor / AutoLunchManager / NotificationOrchestrator
  • update_display 변화 감지 — 직전 값과 동일 시 setText 스킵 (1Hz hot-path 부하↓)
  • 언어 변경 시 자동 재시작 제안 — 사용자 동의 후 os.execv로 재시작
  • CHANGELOG.md + GitHub Actions CI 추가

Changed

  • 5분 throttle 알림 (건강/주간/누적) _last_5min_bucket 명시 가드
  • PyInstaller excludespandas, numpy.testing, PyQt5.QtWebEngineWidgets 추가

[2.0.0] — 2026-04-29

Added

  • 단축근무 지원: work_minutes(분 단위) + 5종 프리셋 (8h / 7h30m / 7h / 6h / 4h)
  • 자동 점심시간 적용 (출근 4h 경과)
  • DB 자동 백업 (~/.clockout_backups/, 7일 회전, SQLite backup API)
  • 화면 잠금 자동 외출/복귀
  • 공휴일 자동 등록 (holidays 패키지, 음력 명절 포함)
  • 시스템 트레이 완전 통합 (점심/외출/통계/캘린더/도움말)
  • 미니 위젯 (Always-on-top, 드래그 가능)
  • matplotlib 차트 (일별/요일별 근무시간)
  • 건강·주간·누적 알림 (3종 추가)
  • i18n 인프라 (한국어/English, 28 카테고리, HelpView ko/en HTML)
  • 설정 키 상수 모듈 (core/settings_keys.py)
  • DB 경로 override (클라우드 동기화 폴더 지정 가능)
  • 앱 단축키 7종 (Ctrl+O/L/D/B, F1, Ctrl+R, Ctrl+,)
  • 사용 설명 가이드 (HelpView, 6 탭)
  • 48 통합 시나리오 + 5 i18n GUI + 8 widget smoke 테스트

Fixed

  • +29h remaining time bugbreak_minutes에서 overtime 차감하던 로직 수정
  • annual_leave_total / annual_leave_days 키 불일치 — 양방향 자동 동기화

[1.0.0] — 2026-01-09

  • 최초 릴리스: 자동 출퇴근 / 점심 관리 / 30분 단위 연장근무 적립 / 캘린더 / 통계