""" 도전과제 시스템 — 357개 자동 평가. 설계: - 각 도전과제는 dataclass `Achievement` + 평가 함수 (db) -> (progress, target). - 평가자는 5분 throttle로 호출되어 미획득 도전과제만 검사. - 신규 잠금 해제 시 main_window의 `notify_achievement_unlocked` 시그널 emit. - 진행도(progress)는 부분 달성 표시용 — UI 게이지에 활용. 데이터 소스: - work_records: 출근/퇴근 기록 - overtime_bank, overtime_usage: 적립/사용 - leave_records: 연차 - break_records: 외출/식사 - holidays: 공휴일 - settings: 사용자 메타 + 뷰 카운터 - notification_log: 휴식 권고 카운트 """ from __future__ import annotations from core.i18n import tr from dataclasses import dataclass, field from datetime import date, datetime, timedelta from typing import Callable, Optional, List, Tuple # === 등급 상수 === TIER_BRONZE = 'bronze' TIER_SILVER = 'silver' TIER_GOLD = 'gold' TIER_PLATINUM = 'platinum' TIER_LEGEND = 'legend' # === 카테고리 === CAT_STREAK = 'streak' CAT_PUNCTUAL = 'punctual' CAT_BALANCE = 'balance' CAT_OT_BANK = 'ot_bank' CAT_OT_USE = 'ot_use' CAT_LEAVE = 'leave' CAT_HEALTH = 'health' CAT_SPECIAL_DAY = 'special_day' CAT_PATTERN = 'pattern' CAT_MILESTONE = 'milestone' CAT_SEASON = 'season' CAT_TIME_SLOT = 'time_slot' CAT_MEAL = 'meal' CAT_BREAK = 'break_use' CAT_SETTINGS = 'settings' CAT_STATS = 'stats' CAT_SECRET = 'secret' CAT_KOREA = 'korea' CAT_AMBITION = 'ambition' CAT_META = 'meta' @dataclass class Achievement: """도전과제 정의. 평가 함수 `evaluator`는 `(db) -> (progress, target)` 시그니처. progress >= target이면 잠금 해제. progress가 0이면 미시작. """ code: str name: str description: str category: str tier: str badge_icon: str is_secret: bool = False target: int = 1 evaluator: Optional[Callable] = field(default=None, repr=False) # ============================================================ # 평가 헬퍼 # ============================================================ def _count_work_records(db) -> int: """전체 work_records 수 (clock_in이 있는 모든 행).""" with db._conn() as conn: cur = conn.cursor() cur.execute("SELECT COUNT(*) FROM work_records") return cur.fetchone()[0] def _count_clocked_out(db) -> int: """퇴근까지 마친 work_records 수.""" with db._conn() as conn: cur = conn.cursor() cur.execute("SELECT COUNT(*) FROM work_records WHERE clock_out IS NOT NULL") return cur.fetchone()[0] def _consecutive_workdays(db) -> int: """오늘 또는 마지막 출근일 기준 연속 영업일 출근 수. 영업일 = 토/일이 아니고 holidays 테이블에 없는 날. """ today = date.today() with db._conn() as conn: cur = conn.cursor() cur.execute("SELECT date FROM work_records ORDER BY date DESC LIMIT 1") row = cur.fetchone() if not row: return 0 last = datetime.strptime(row[0], '%Y-%m-%d').date() if (today - last).days > 3: return 0 # 끊김 # holidays 셋 cur.execute("SELECT date FROM holidays") holidays = {r[0] for r in cur.fetchall()} cur.execute("SELECT date FROM work_records") worked = {r[0] for r in cur.fetchall()} streak = 0 d = last while True: is_workday = d.weekday() < 5 and d.isoformat() not in holidays if is_workday: if d.isoformat() in worked: streak += 1 else: break d -= timedelta(days=1) if streak > 1000: # safety break return streak def _consecutive_calendar_days(db) -> int: """달력일 기준 연속 출근 수 (주말 포함).""" with db._conn() as conn: cur = conn.cursor() cur.execute("SELECT date FROM work_records ORDER BY date DESC") dates = [datetime.strptime(r[0], '%Y-%m-%d').date() for r in cur.fetchall()] if not dates: return 0 streak = 1 for i in range(1, len(dates)): if (dates[i-1] - dates[i]).days == 1: streak += 1 else: break return streak def _ot_balance_minutes(db) -> int: """현재 연장근무 잔액 (분).""" return db.get_total_overtime_balance() def _ot_total_earned(db) -> int: """누적 적립 분.""" with db._conn() as conn: cur = conn.cursor() cur.execute("SELECT COALESCE(SUM(earned_minutes), 0) FROM overtime_bank") return cur.fetchone()[0] def _ot_total_used(db) -> int: """누적 사용 분.""" with db._conn() as conn: cur = conn.cursor() cur.execute("SELECT COALESCE(SUM(used_minutes), 0) FROM overtime_usage") return cur.fetchone()[0] def _count_punctual_clockouts(db) -> int: """정시 퇴근 (overtime_minutes <= 0) 횟수.""" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT COUNT(*) FROM work_records WHERE clock_out IS NOT NULL AND COALESCE(overtime_minutes, 0) <= 0 """) return cur.fetchone()[0] def _count_clock_in_before(db, hour: int, minute: int = 0) -> int: """특정 시각 이전 출근 횟수.""" threshold = f"{hour:02d}:{minute:02d}:00" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT COUNT(*) FROM work_records WHERE clock_in < ? """, (threshold,)) return cur.fetchone()[0] def _count_clock_in_in_range(db, start_hh: int, end_hh: int) -> int: """[start_hh:00, end_hh:00) 시간대 출근 횟수.""" s = f"{start_hh:02d}:00:00" e = f"{end_hh:02d}:00:00" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT COUNT(*) FROM work_records WHERE clock_in >= ? AND clock_in < ? """, (s, e)) return cur.fetchone()[0] def _count_clock_out_after(db, hour: int) -> int: """특정 시각(시) 이후 퇴근 횟수. 자정 이후는 hour=24로 별도 처리.""" threshold = f"{hour:02d}:00:00" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT COUNT(*) FROM work_records WHERE clock_out IS NOT NULL AND clock_out >= ? """, (threshold,)) return cur.fetchone()[0] def _count_clock_out_after_midnight(db) -> int: """자정 이후 ~ 오전 퇴근 횟수 (clock_out < clock_in인 경우, 익일). 같은 날짜 내에서 clock_out HH:MM:SS가 clock_in보다 작으면 자정 넘긴 것.""" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT COUNT(*) FROM work_records WHERE clock_out IS NOT NULL AND clock_out < clock_in """) return cur.fetchone()[0] def _count_weekend_clockins(db) -> int: """토/일 출근 횟수.""" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT COUNT(*) FROM work_records WHERE CAST(strftime('%w', date) AS INTEGER) IN (0, 6) """) return cur.fetchone()[0] def _count_holiday_clockins(db) -> int: """공휴일 출근 (holidays 테이블에 있는 날).""" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT COUNT(*) FROM work_records w WHERE EXISTS (SELECT 1 FROM holidays h WHERE h.date = w.date) """) return cur.fetchone()[0] def _has_clockin_on(db, mm_dd: str) -> bool: """특정 MM-DD에 출근 기록 있는지.""" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT 1 FROM work_records WHERE strftime('%m-%d', date) = ? LIMIT 1 """, (mm_dd,)) return cur.fetchone() is not None def _has_punctual_clockout_on(db, mm_dd: str) -> bool: """특정 MM-DD에 정시 퇴근.""" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT 1 FROM work_records WHERE strftime('%m-%d', date) = ? AND clock_out IS NOT NULL AND COALESCE(overtime_minutes, 0) <= 0 LIMIT 1 """, (mm_dd,)) return cur.fetchone() is not None def _count_lunch_registrations(db) -> int: with db._conn() as conn: cur = conn.cursor() cur.execute("SELECT COUNT(*) FROM work_records WHERE lunch_break = 1") return cur.fetchone()[0] def _count_dinner_registrations(db) -> int: with db._conn() as conn: cur = conn.cursor() cur.execute("SELECT COUNT(*) FROM work_records WHERE dinner_break = 1") return cur.fetchone()[0] def _count_break_records_type(db, break_type: str = 'break') -> int: with db._conn() as conn: cur = conn.cursor() cur.execute("SELECT COUNT(*) FROM break_records WHERE break_type = ?", (break_type,)) return cur.fetchone()[0] def _count_leave_records(db, leave_type: str = None) -> int: with db._conn() as conn: cur = conn.cursor() if leave_type: cur.execute("SELECT COUNT(*) FROM leave_records WHERE leave_type = ?", (leave_type,)) else: cur.execute("SELECT COUNT(*) FROM leave_records") return cur.fetchone()[0] def _has_leave_with_days(db, days_value: float) -> bool: """특정 days 값(0.5, 0.25 등)의 연차 사용 여부.""" with db._conn() as conn: cur = conn.cursor() cur.execute("SELECT 1 FROM leave_records WHERE days = ? LIMIT 1", (days_value,)) return cur.fetchone() is not None def _consecutive_leave_days(db) -> int: """가장 긴 연속 연차 사용 일수.""" with db._conn() as conn: cur = conn.cursor() cur.execute("SELECT date FROM leave_records ORDER BY date") dates = [datetime.strptime(r[0], '%Y-%m-%d').date() for r in cur.fetchall()] if not dates: return 0 max_streak = 1 cur_streak = 1 for i in range(1, len(dates)): if (dates[i] - dates[i-1]).days == 1: cur_streak += 1 max_streak = max(max_streak, cur_streak) else: cur_streak = 1 return max_streak def _setting_int(db, key: str, default: int = 0) -> int: return db.get_setting_int(key, default) def _setting_str(db, key: str, default: str = '') -> str: return db.get_setting(key, default) or default def _days_since_first_work(db) -> int: """첫 work_records로부터 오늘까지 경과 일수.""" hire = _setting_str(db, 'hire_date', '') if not hire: return 0 try: d = datetime.strptime(hire, '%Y-%m-%d').date() return (date.today() - d).days except ValueError: return 0 def _has_clockin_on_date(db, target_date: date) -> bool: with db._conn() as conn: cur = conn.cursor() cur.execute("SELECT 1 FROM work_records WHERE date = ? LIMIT 1", (target_date.isoformat(),)) return cur.fetchone() is not None def _count_overtime_days(db, min_minutes: int = 1) -> int: """야근(overtime_minutes >= min_minutes)한 일수.""" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT COUNT(*) FROM work_records WHERE overtime_minutes >= ? """, (min_minutes,)) return cur.fetchone()[0] def _count_clockouts_at_minute(db, minute: int) -> int: """퇴근 시각의 분 자릿수가 특정 값인 횟수.""" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT COUNT(*) FROM work_records WHERE clock_out IS NOT NULL AND CAST(strftime('%M', clock_out) AS INTEGER) = ? """, (minute,)) return cur.fetchone()[0] def _count_in_year_month(db, year: int, month: int) -> int: """특정 연-월 출근 일수.""" prefix = f"{year:04d}-{month:02d}" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT COUNT(*) FROM work_records WHERE date LIKE ? || '%' """, (prefix,)) return cur.fetchone()[0] def _workdays_in_year_month(year: int, month: int) -> int: """해당 월의 영업일 수 (토/일 제외, 공휴일은 무시).""" from calendar import monthrange last = monthrange(year, month)[1] count = 0 for d in range(1, last + 1): if date(year, month, d).weekday() < 5: count += 1 return count # ============================================================ # 도전과제 정의 (357개) # ============================================================ def _make_streak_eval(target_days: int, business_only: bool = True): def _eval(db): cur = (_consecutive_workdays(db) if business_only else _consecutive_calendar_days(db)) return min(cur, target_days), target_days return _eval def _make_count_eval(getter, target: int): def _eval(db): cur = getter(db) return min(cur, target), target return _eval def _bool_eval(condition_fn): """True/False 조건 → (1, 1) 또는 (0, 1).""" def _eval(db): return (1, 1) if condition_fn(db) else (0, 1) return _eval # ---- 1. 출근 streak (24개 — 22번 거북이 제거) ---- _STREAK_DEFS = [ # (code, name, desc, target, evaluator, tier, icon) ('streak_first', tr('achieve.streak_first.name'), tr('achieve.streak_first.desc'), 1, _bool_eval(lambda db: _count_work_records(db) >= 1), TIER_BRONZE, '👋'), ('streak_3', tr('achieve.streak_3.name'), tr('achieve.streak_3.desc'), 3, _make_streak_eval(3), TIER_BRONZE, '🌱'), ('streak_5', tr('achieve.streak_5.name'), tr('achieve.streak_5.desc'), 5, _make_streak_eval(5), TIER_SILVER, '📅'), ('streak_7_cal', tr('achieve.streak_7_cal.name'), tr('achieve.streak_7_cal.desc'), 7, _make_streak_eval(7, business_only=False), TIER_SILVER, '🔥'), ('streak_10', tr('achieve.streak_10.name'), tr('achieve.streak_10.desc'), 10, _make_streak_eval(10), TIER_SILVER, '💪'), ('streak_22', tr('achieve.streak_22.name'), tr('achieve.streak_22.desc'), 22, _make_streak_eval(22), TIER_GOLD, '🏔️'), ('streak_50', tr('achieve.streak_50.name'), tr('achieve.streak_50.desc'), 50, _make_streak_eval(50), TIER_GOLD, '🎯'), ('streak_100', tr('achieve.streak_100.name'), tr('achieve.streak_100.desc'), 100, _make_streak_eval(100), TIER_PLATINUM, '💎'), ('streak_quarter', tr('achieve.streak_quarter.name'), tr('achieve.streak_quarter.desc'), 65, _make_streak_eval(65), TIER_PLATINUM, '🏆'), ('streak_half_year', tr('achieve.streak_half_year.name'), tr('achieve.streak_half_year.desc'), 130, _make_streak_eval(130), TIER_PLATINUM, '👑'), ('streak_year', tr('achieve.streak_year.name'), tr('achieve.streak_year.desc'), 260, _make_streak_eval(260), TIER_LEGEND, '🌟'), ('streak_200', tr('achieve.streak_200.name'), tr('achieve.streak_200.desc'), 200, _make_streak_eval(200), TIER_LEGEND, '🌌'), ('streak_365_cal', tr('achieve.streak_365_cal.name'), tr('achieve.streak_365_cal.desc'), 365, _make_streak_eval(365, business_only=False), TIER_LEGEND, '🛡️'), ('streak_resilience', tr('achieve.streak_resilience.name'), tr('achieve.streak_resilience.desc'), 1, _bool_eval(lambda db: _consecutive_workdays(db) >= 1 and _count_work_records(db) >= 5), TIER_BRONZE, '⚡'), ('streak_total_100', tr('achieve.streak_total_100.name'), tr('achieve.streak_total_100.desc'), 100, _make_count_eval(_count_work_records, 100), TIER_GOLD, '💼'), ('streak_total_500', tr('achieve.streak_total_500.name'), tr('achieve.streak_total_500.desc'), 500, _make_count_eval(_count_work_records, 500), TIER_PLATINUM, '🏛️'), ('streak_total_1000', tr('achieve.streak_total_1000.name'), tr('achieve.streak_total_1000.desc'), 1000, _make_count_eval(_count_work_records, 1000), TIER_LEGEND, '🎖️'), ] def _count_weekday_clockins(db, weekday: int) -> int: """특정 요일(0=월 ... 6=일) 출근 횟수.""" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT COUNT(*) FROM work_records WHERE CAST(strftime('%w', date) AS INTEGER) = ? """, (weekday,)) return cur.fetchone()[0] _STREAK_DEFS.extend([ ('streak_monday_10', tr('achieve.streak_monday_10.name'), tr('achieve.streak_monday_10.desc'), 10, _make_count_eval(lambda db: _count_weekday_clockins(db, 1), 10), TIER_SILVER, '🌅'), ('streak_friday_10', tr('achieve.streak_friday_10.name'), tr('achieve.streak_friday_10.desc'), 10, _make_count_eval(lambda db: _count_weekday_clockins(db, 5), 10), TIER_SILVER, '🌒'), ]) # ---- 2. 시간 엄수 (19개 - 34/46 제거) ---- _PUNCTUAL_DEFS = [ ('punc_before_8_1', tr('achieve.punc_before_8_1.name'), tr('achieve.punc_before_8_1.desc'), 1, _make_count_eval(lambda db: _count_clock_in_before(db, 8), 1), TIER_BRONZE, '🌄'), ('punc_before_8_10', tr('achieve.punc_before_8_10.name'), tr('achieve.punc_before_8_10.desc'), 10, _make_count_eval(lambda db: _count_clock_in_before(db, 8), 10), TIER_SILVER, '🐦'), ('punc_before_8_30', tr('achieve.punc_before_8_30.name'), tr('achieve.punc_before_8_30.desc'), 30, _make_count_eval(lambda db: _count_clock_in_before(db, 8), 30), TIER_GOLD, '🌞'), ('punc_before_6_1', tr('achieve.punc_before_6_1.name'), tr('achieve.punc_before_6_1.desc'), 1, _make_count_eval(lambda db: _count_clock_in_before(db, 6), 1), TIER_GOLD, '🥱'), ('punc_before_6_10', tr('achieve.punc_before_6_10.name'), tr('achieve.punc_before_6_10.desc'), 10, _make_count_eval(lambda db: _count_clock_in_before(db, 6), 10), TIER_PLATINUM, '🌑'), ('punc_before_5', tr('achieve.punc_before_5.name'), tr('achieve.punc_before_5.desc'), 1, _make_count_eval(lambda db: _count_clock_in_before(db, 5), 1), TIER_LEGEND, '🌌'), ('punc_at_9', tr('achieve.punc_at_9.name'), tr('achieve.punc_at_9.desc'), 1, _make_count_eval(lambda db: _count_clock_in_in_range_minute(db, 9, 0, 9, 2), 1), TIER_BRONZE, '🎯'), ('punc_at_9_5', tr('achieve.punc_at_9_5.name'), tr('achieve.punc_at_9_5.desc'), 5, _make_count_eval(lambda db: _count_clock_in_in_range_minute(db, 9, 0, 9, 2), 5), TIER_SILVER, '🏹'), ('punc_late_5min', tr('achieve.punc_late_5min.name'), tr('achieve.punc_late_5min.desc'), 1, _make_count_eval(lambda db: _count_clock_in_in_range_minute(db, 9, 0, 9, 6), 1), TIER_BRONZE, '🛌'), ('punc_at_909', tr('achieve.punc_at_909.name'), tr('achieve.punc_at_909.desc'), 1, _make_count_eval(lambda db: _count_clock_in_in_range_minute(db, 9, 9, 9, 10), 1), TIER_GOLD, '🎰'), ] def _count_clock_in_in_range_minute(db, sh: int, sm: int, eh: int, em: int) -> int: s = f"{sh:02d}:{sm:02d}:00" e = f"{eh:02d}:{em:02d}:00" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT COUNT(*) FROM work_records WHERE clock_in >= ? AND clock_in < ? """, (s, e)) return cur.fetchone()[0] # ---- 3. 워라밸·정시 퇴근 (8개 코어) ---- _BALANCE_DEFS = [ ('bal_first_punct', tr('achieve.bal_first_punct.name'), tr('achieve.bal_first_punct.desc'), 1, _make_count_eval(_count_punctual_clockouts, 1), TIER_BRONZE, '🚪'), ('bal_punct_10', tr('achieve.bal_punct_10.name'), tr('achieve.bal_punct_10.desc'), 10, _make_count_eval(_count_punctual_clockouts, 10), TIER_SILVER, '🎉'), ('bal_punct_30', tr('achieve.bal_punct_30.name'), tr('achieve.bal_punct_30.desc'), 30, _make_count_eval(_count_punctual_clockouts, 30), TIER_GOLD, '🏃'), ('bal_punct_100', tr('achieve.bal_punct_100.name'), tr('achieve.bal_punct_100.desc'), 100, _make_count_eval(_count_punctual_clockouts, 100), TIER_LEGEND, '🏖️'), ('bal_punct_300', tr('achieve.bal_punct_300.name'), tr('achieve.bal_punct_300.desc'), 300, _make_count_eval(_count_punctual_clockouts, 300), TIER_LEGEND, '🪐'), ] # ---- 4. 연장근무 적립 ---- _OT_BANK_DEFS = [ ('ot_first_30m', tr('achieve.ot_first_30m.name'), tr('achieve.ot_first_30m.desc'), 30, _make_count_eval(_ot_total_earned, 30), TIER_BRONZE, '💰'), ('ot_total_60m', tr('achieve.ot_total_60m.name'), tr('achieve.ot_total_60m.desc'), 60, _make_count_eval(_ot_total_earned, 60), TIER_BRONZE, '💵'), ('ot_total_5h', tr('achieve.ot_total_5h.name'), tr('achieve.ot_total_5h.desc'), 300, _make_count_eval(_ot_total_earned, 300), TIER_SILVER, '🏦'), ('ot_total_10h', tr('achieve.ot_total_10h.name'), tr('achieve.ot_total_10h.desc'), 600, _make_count_eval(_ot_total_earned, 600), TIER_SILVER, '💎'), ('ot_total_25h', tr('achieve.ot_total_25h.name'), tr('achieve.ot_total_25h.desc'), 1500, _make_count_eval(_ot_total_earned, 1500), TIER_GOLD, '🏆'), ('ot_total_50h', tr('achieve.ot_total_50h.name'), tr('achieve.ot_total_50h.desc'), 3000, _make_count_eval(_ot_total_earned, 3000), TIER_GOLD, '🎯'), ('ot_total_100h', tr('achieve.ot_total_100h.name'), tr('achieve.ot_total_100h.desc'), 6000, _make_count_eval(_ot_total_earned, 6000), TIER_PLATINUM, '🏔️'), ('ot_total_200h', tr('achieve.ot_total_200h.name'), tr('achieve.ot_total_200h.desc'), 12000, _make_count_eval(_ot_total_earned, 12000), TIER_PLATINUM, '🌑'), ('ot_total_300h', tr('achieve.ot_total_300h.name'), tr('achieve.ot_total_300h.desc'), 18000, _make_count_eval(_ot_total_earned, 18000), TIER_LEGEND, '⚠️'), ('ot_total_500h', tr('achieve.ot_total_500h.name'), tr('achieve.ot_total_500h.desc'), 30000, _make_count_eval(_ot_total_earned, 30000), TIER_LEGEND, '🚑'), ] # ---- 5. 연장근무 사용 ---- _OT_USE_DEFS = [ ('use_first', tr('achieve.use_first.name'), tr('achieve.use_first.desc'), 1, _bool_eval(lambda db: _ot_total_used(db) > 0), TIER_BRONZE, '🛌'), ('use_total_5h', tr('achieve.use_total_5h.name'), tr('achieve.use_total_5h.desc'), 300, _make_count_eval(_ot_total_used, 300), TIER_SILVER, '🎁'), ('use_total_25h', tr('achieve.use_total_25h.name'), tr('achieve.use_total_25h.desc'), 1500, _make_count_eval(_ot_total_used, 1500), TIER_GOLD, '🛀'), ('use_total_50h', tr('achieve.use_total_50h.name'), tr('achieve.use_total_50h.desc'), 3000, _make_count_eval(_ot_total_used, 3000), TIER_GOLD, '🏖️'), ('use_total_100h', tr('achieve.use_total_100h.name'), tr('achieve.use_total_100h.desc'), 6000, _make_count_eval(_ot_total_used, 6000), TIER_PLATINUM, '💆'), ] # ---- 6. 연차 ---- _LEAVE_DEFS = [ ('leave_first', tr('achieve.leave_first.name'), tr('achieve.leave_first.desc'), 1, _make_count_eval(_count_leave_records, 1), TIER_BRONZE, '🌴'), ('leave_half', tr('achieve.leave_half.name'), tr('achieve.leave_half.desc'), 1, _bool_eval(lambda db: _has_leave_with_days(db, 0.5)), TIER_BRONZE, '🍃'), ('leave_quarter', tr('achieve.leave_quarter.name'), tr('achieve.leave_quarter.desc'), 1, _bool_eval(lambda db: _has_leave_with_days(db, 0.25)), TIER_BRONZE, '⏱️'), ('leave_streak_3', tr('achieve.leave_streak_3.name'), tr('achieve.leave_streak_3.desc'), 3, _make_count_eval(_consecutive_leave_days, 3), TIER_SILVER, '🏝️'), ('leave_streak_5', tr('achieve.leave_streak_5.name'), tr('achieve.leave_streak_5.desc'), 5, _make_count_eval(_consecutive_leave_days, 5), TIER_GOLD, '🌅'), ('leave_streak_7', tr('achieve.leave_streak_7.name'), tr('achieve.leave_streak_7.desc'), 7, _make_count_eval(_consecutive_leave_days, 7), TIER_PLATINUM, '🛬'), ('leave_total_10', tr('achieve.leave_total_10.name'), tr('achieve.leave_total_10.desc'), 10, _make_count_eval(_count_leave_records, 10), TIER_SILVER, '🌊'), ('leave_sick', tr('achieve.leave_sick.name'), tr('achieve.leave_sick.desc'), 1, _make_count_eval(lambda db: _count_leave_records(db, 'sick'), 1), TIER_BRONZE, '🏥'), ] # ---- 7. 식사 (점심/저녁) ---- _MEAL_DEFS = [ ('meal_lunch_first', tr('achieve.meal_lunch_first.name'), tr('achieve.meal_lunch_first.desc'), 1, _make_count_eval(_count_lunch_registrations, 1), TIER_BRONZE, '🍱'), ('meal_lunch_30', tr('achieve.meal_lunch_30.name'), tr('achieve.meal_lunch_30.desc'), 30, _make_count_eval(_count_lunch_registrations, 30), TIER_SILVER, '🥢'), ('meal_lunch_100', tr('achieve.meal_lunch_100.name'), tr('achieve.meal_lunch_100.desc'), 100, _make_count_eval(_count_lunch_registrations, 100), TIER_GOLD, '🍜'), ('meal_dinner_first', tr('achieve.meal_dinner_first.name'), tr('achieve.meal_dinner_first.desc'), 1, _make_count_eval(_count_dinner_registrations, 1), TIER_BRONZE, '🍽️'), ('meal_dinner_10', tr('achieve.meal_dinner_10.name'), tr('achieve.meal_dinner_10.desc'), 10, _make_count_eval(_count_dinner_registrations, 10), TIER_SILVER, '🍛'), ('meal_dinner_30', tr('achieve.meal_dinner_30.name'), tr('achieve.meal_dinner_30.desc'), 30, _make_count_eval(_count_dinner_registrations, 30), TIER_GOLD, '🌃'), ('meal_lunch_actual', tr('achieve.meal_lunch_actual.name'), tr('achieve.meal_lunch_actual.desc'), 1, _make_count_eval(lambda db: _count_break_records_type(db, 'lunch'), 1), TIER_BRONZE, '⏱️'), ('meal_dinner_actual', tr('achieve.meal_dinner_actual.name'), tr('achieve.meal_dinner_actual.desc'), 1, _make_count_eval(lambda db: _count_break_records_type(db, 'dinner'), 1), TIER_BRONZE, '⏰'), ] # ---- 8. 외출 ---- _BREAK_DEFS = [ ('break_first', tr('achieve.break_first.name'), tr('achieve.break_first.desc'), 1, _make_count_eval(lambda db: _count_break_records_type(db, 'break'), 1), TIER_BRONZE, '🚶'), ('break_10', tr('achieve.break_10.name'), tr('achieve.break_10.desc'), 10, _make_count_eval(lambda db: _count_break_records_type(db, 'break'), 10), TIER_SILVER, '🚪'), ('break_50', tr('achieve.break_50.name'), tr('achieve.break_50.desc'), 50, _make_count_eval(lambda db: _count_break_records_type(db, 'break'), 50), TIER_GOLD, '🚶‍♂️'), ] # ---- 9. 시간대별 ---- _TIME_SLOT_DEFS = [ ('slot_in_06', tr('achieve.slot_in_06.name'), tr('achieve.slot_in_06.desc'), 1, _make_count_eval(lambda db: _count_clock_in_in_range(db, 6, 7), 1), TIER_BRONZE, '🌅'), ('slot_in_07', tr('achieve.slot_in_07.name'), tr('achieve.slot_in_07.desc'), 1, _make_count_eval(lambda db: _count_clock_in_in_range(db, 7, 8), 1), TIER_BRONZE, '🌄'), ('slot_in_08', tr('achieve.slot_in_08.name'), tr('achieve.slot_in_08.desc'), 1, _make_count_eval(lambda db: _count_clock_in_in_range(db, 8, 9), 1), TIER_BRONZE, '☀️'), ('slot_in_10', tr('achieve.slot_in_10.name'), tr('achieve.slot_in_10.desc'), 1, _make_count_eval(lambda db: _count_clock_in_in_range(db, 10, 11), 1), TIER_BRONZE, '🕙'), ('slot_in_11', tr('achieve.slot_in_11.name'), tr('achieve.slot_in_11.desc'), 1, _make_count_eval(lambda db: _count_clock_in_in_range(db, 11, 12), 1), TIER_SILVER, '🕦'), ('slot_out_19', tr('achieve.slot_out_19.name'), tr('achieve.slot_out_19.desc'), 10, _make_count_eval(lambda db: _count_clockouts_in_hour(db, 19), 10), TIER_SILVER, '🌆'), ('slot_out_20', tr('achieve.slot_out_20.name'), tr('achieve.slot_out_20.desc'), 10, _make_count_eval(lambda db: _count_clockouts_in_hour(db, 20), 10), TIER_GOLD, '🌌'), ('slot_out_21', tr('achieve.slot_out_21.name'), tr('achieve.slot_out_21.desc'), 5, _make_count_eval(lambda db: _count_clockouts_in_hour(db, 21), 5), TIER_GOLD, '🌑'), ('slot_out_22', tr('achieve.slot_out_22.name'), tr('achieve.slot_out_22.desc'), 1, _make_count_eval(lambda db: _count_clockouts_in_hour(db, 22), 1), TIER_PLATINUM, '🦉'), ('slot_out_23', tr('achieve.slot_out_23.name'), tr('achieve.slot_out_23.desc'), 1, _make_count_eval(lambda db: _count_clockouts_in_hour(db, 23), 1), TIER_PLATINUM, '🦇'), ('slot_midnight', tr('achieve.slot_midnight.name'), tr('achieve.slot_midnight.desc'), 1, _make_count_eval(_count_clock_out_after_midnight, 1), TIER_LEGEND, '🌚'), ('slot_midnight_3', tr('achieve.slot_midnight_3.name'), tr('achieve.slot_midnight_3.desc'), 3, _make_count_eval(_count_clock_out_after_midnight, 3), TIER_LEGEND, '🌌'), ] def _count_clockouts_in_hour(db, hour: int) -> int: """clock_out이 hour시대(HH:00-HH:59)인 횟수. 자정 넘김 케이스 무시.""" s = f"{hour:02d}:00:00" e = f"{hour+1:02d}:00:00" if hour < 23 else "23:59:59" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT COUNT(*) FROM work_records WHERE clock_out IS NOT NULL AND clock_out >= ? AND clock_out < ? AND clock_out >= clock_in """, (s, e)) return cur.fetchone()[0] # ---- 10. 공휴일·주말 ---- _SPECIAL_DAY_DEFS = [ ('weekend_1', tr('achieve.weekend_1.name'), tr('achieve.weekend_1.desc'), 1, _make_count_eval(_count_weekend_clockins, 1), TIER_SILVER, '🌃'), ('weekend_5', tr('achieve.weekend_5.name'), tr('achieve.weekend_5.desc'), 5, _make_count_eval(_count_weekend_clockins, 5), TIER_GOLD, '🌑'), ('weekend_20', tr('achieve.weekend_20.name'), tr('achieve.weekend_20.desc'), 20, _make_count_eval(_count_weekend_clockins, 20), TIER_PLATINUM, '💀'), ('holiday_1', tr('achieve.holiday_1.name'), tr('achieve.holiday_1.desc'), 1, _make_count_eval(_count_holiday_clockins, 1), TIER_GOLD, '📆'), ('holiday_5', tr('achieve.holiday_5.name'), tr('achieve.holiday_5.desc'), 5, _make_count_eval(_count_holiday_clockins, 5), TIER_LEGEND, '⚠️'), ('day_christmas', tr('achieve.day_christmas.name'), tr('achieve.day_christmas.desc'), 1, _bool_eval(lambda db: _has_clockin_on(db, '12-25')), TIER_GOLD, '🎄'), ('day_newyear', tr('achieve.day_newyear.name'), tr('achieve.day_newyear.desc'), 1, _bool_eval(lambda db: _has_clockin_on(db, '01-01')), TIER_GOLD, '🎊'), ('day_liberation', tr('achieve.day_liberation.name'), tr('achieve.day_liberation.desc'), 1, _bool_eval(lambda db: _has_clockin_on(db, '08-15')), TIER_SILVER, '🎆'), ('day_children', tr('achieve.day_children.name'), tr('achieve.day_children.desc'), 1, _bool_eval(lambda db: _has_clockin_on(db, '05-05')), TIER_GOLD, '🎀'), ('day_hangul', tr('achieve.day_hangul.name'), tr('achieve.day_hangul.desc'), 1, _bool_eval(lambda db: _has_clockin_on(db, '10-09')), TIER_SILVER, '🎤'), ('day_valentine', tr('achieve.day_valentine.name'), tr('achieve.day_valentine.desc'), 1, _bool_eval(lambda db: _has_clockin_on(db, '02-14')), TIER_BRONZE, '💝'), ('day_white', tr('achieve.day_white.name'), tr('achieve.day_white.desc'), 1, _bool_eval(lambda db: _has_clockin_on(db, '03-14')), TIER_BRONZE, '🌹'), ('day_pepero', tr('achieve.day_pepero.name'), tr('achieve.day_pepero.desc'), 1, _bool_eval(lambda db: _has_clockin_on(db, '11-11')), TIER_SILVER, '🍫'), ('day_halloween', tr('achieve.day_halloween.name'), tr('achieve.day_halloween.desc'), 1, _bool_eval(lambda db: _has_clockin_on(db, '10-31')), TIER_BRONZE, '🎃'), ('day_aprilfools', tr('achieve.day_aprilfools.name'), tr('achieve.day_aprilfools.desc'), 1, _bool_eval(lambda db: _has_clockin_on(db, '04-01')), TIER_BRONZE, '🃏'), ('day_77', tr('achieve.day_77.name'), tr('achieve.day_77.desc'), 1, _bool_eval(lambda db: _has_clockin_on(db, '07-07')), TIER_SILVER, '🎋'), ('day_dongji', tr('achieve.day_dongji.name'), tr('achieve.day_dongji.desc'), 1, _bool_eval(lambda db: _has_clockin_on(db, '12-22')), TIER_BRONZE, '🎇'), ('day_parents', tr('achieve.day_parents.name'), tr('achieve.day_parents.desc'), 1, _bool_eval(lambda db: _has_punctual_clockout_on(db, '05-08')), TIER_SILVER, '🪅'), ('day_teacher', tr('achieve.day_teacher.name'), tr('achieve.day_teacher.desc'), 1, _bool_eval(lambda db: _has_punctual_clockout_on(db, '05-15')), TIER_BRONZE, '🎂'), ('day_xmas_eve', tr('achieve.day_xmas_eve.name'), tr('achieve.day_xmas_eve.desc'), 1, _bool_eval(lambda db: _has_punctual_clockout_on(db, '12-24')), TIER_SILVER, '🎁'), ('day_earth', tr('achieve.day_earth.name'), tr('achieve.day_earth.desc'), 1, _bool_eval(lambda db: _has_clockin_on(db, '04-22')), TIER_GOLD, '🌏'), ] # ---- 11. 시즌·월별 ---- def _make_month_full_attendance_eval(month: int): """해당 월 영업일 모두 출근.""" def _eval(db): year = date.today().year target = _workdays_in_year_month(year, month) cur = _count_in_year_month(db, year, month) # 영업일 수 정확히 카운트는 holidays 제외 안 함 — 단순화 return min(cur, target), max(target, 1) return _eval def _make_month_first_eval(month: int): def _eval(db): year = date.today().year return ((1, 1) if _count_in_year_month(db, year, month) >= 1 else (0, 1)) return _eval _SEASON_DEFS = [ ('season_jan', tr('achieve.season_jan.name'), tr('achieve.season_jan.desc'), 1, _make_month_first_eval(1), TIER_BRONZE, '⛄'), ('season_feb', tr('achieve.season_feb.name'), tr('achieve.season_feb.desc'), 1, _make_month_full_attendance_eval(2), TIER_SILVER, '🌨️'), ('season_mar', tr('achieve.season_mar.name'), tr('achieve.season_mar.desc'), 1, _make_month_first_eval(3), TIER_BRONZE, '🌸'), ('season_apr', tr('achieve.season_apr.name'), tr('achieve.season_apr.desc'), 1, _make_month_full_attendance_eval(4), TIER_BRONZE, '🌷'), ('season_may', tr('achieve.season_may.name'), tr('achieve.season_may.desc'), 1, _make_month_full_attendance_eval(5), TIER_SILVER, '🌺'), ('season_jun', tr('achieve.season_jun.name'), tr('achieve.season_jun.desc'), 1, _make_month_first_eval(6), TIER_BRONZE, '☀️'), ('season_jul', tr('achieve.season_jul.name'), tr('achieve.season_jul.desc'), 1, _make_month_full_attendance_eval(7), TIER_BRONZE, '🌻'), ('season_aug', tr('achieve.season_aug.name'), tr('achieve.season_aug.desc'), 1, _make_month_full_attendance_eval(8), TIER_SILVER, '🍦'), ('season_sep', tr('achieve.season_sep.name'), tr('achieve.season_sep.desc'), 1, _make_month_first_eval(9), TIER_BRONZE, '🍂'), ('season_oct', tr('achieve.season_oct.name'), tr('achieve.season_oct.desc'), 1, _make_month_full_attendance_eval(10), TIER_BRONZE, '🌾'), ('season_nov', tr('achieve.season_nov.name'), tr('achieve.season_nov.desc'), 1, _make_month_full_attendance_eval(11), TIER_SILVER, '🍁'), ('season_dec', tr('achieve.season_dec.name'), tr('achieve.season_dec.desc'), 1, _make_month_first_eval(12), TIER_BRONZE, '❄️'), ] # ---- 12. 앱 사용 마일스톤 ---- _MILESTONE_DEFS = [ ('mile_first', tr('achieve.mile_first.name'), tr('achieve.mile_first.desc'), 1, _bool_eval(lambda db: _count_work_records(db) >= 1 or _days_since_first_work(db) >= 0), TIER_BRONZE, '👋'), ('mile_7days', tr('achieve.mile_7days.name'), tr('achieve.mile_7days.desc'), 7, _make_count_eval(_days_since_first_work, 7), TIER_BRONZE, '🗓️'), ('mile_30days', tr('achieve.mile_30days.name'), tr('achieve.mile_30days.desc'), 30, _make_count_eval(_days_since_first_work, 30), TIER_SILVER, '📚'), ('mile_365days', tr('achieve.mile_365days.name'), tr('achieve.mile_365days.desc'), 365, _make_count_eval(_days_since_first_work, 365), TIER_PLATINUM, '💎'), ('mile_730days', tr('achieve.mile_730days.name'), tr('achieve.mile_730days.desc'), 730, _make_count_eval(_days_since_first_work, 730), TIER_LEGEND, '🌟'), ('mile_1095days', tr('achieve.mile_1095days.name'), tr('achieve.mile_1095days.desc'), 1095, _make_count_eval(_days_since_first_work, 1095), TIER_LEGEND, '🎖️'), ('mile_5years', tr('achieve.mile_5years.name'), tr('achieve.mile_5years.desc'), 1825, _make_count_eval(_days_since_first_work, 1825), TIER_LEGEND, '🏆'), ('mile_10years', tr('achieve.mile_10years.name'), tr('achieve.mile_10years.desc'), 3650, _make_count_eval(_days_since_first_work, 3650), TIER_LEGEND, '🎖️'), ] # ---- 13. 통계·분석 (view counter 기반) ---- _STATS_DEFS = [ ('stat_weekly_10', tr('achieve.stat_weekly_10.name'), tr('achieve.stat_weekly_10.desc'), 10, _make_count_eval(lambda db: _setting_int(db, 'stat_weekly_view_count'), 10), TIER_BRONZE, '📊'), ('stat_monthly_10', tr('achieve.stat_monthly_10.name'), tr('achieve.stat_monthly_10.desc'), 10, _make_count_eval(lambda db: _setting_int(db, 'stat_monthly_view_count'), 10), TIER_BRONZE, '📈'), ('stat_pattern_10', tr('achieve.stat_pattern_10.name'), tr('achieve.stat_pattern_10.desc'), 10, _make_count_eval(lambda db: _setting_int(db, 'stat_pattern_view_count'), 10), TIER_SILVER, '🔍'), ('stat_calendar_30', tr('achieve.stat_calendar_30.name'), tr('achieve.stat_calendar_30.desc'), 30, _make_count_eval(lambda db: _setting_int(db, 'calendar_view_count'), 30), TIER_SILVER, '📅'), ('stat_report_first', tr('achieve.stat_report_first.name'), tr('achieve.stat_report_first.desc'), 1, _make_count_eval(lambda db: _setting_int(db, 'daily_report_count'), 1), TIER_BRONZE, '📋'), ('stat_report_30', tr('achieve.stat_report_30.name'), tr('achieve.stat_report_30.desc'), 30, _make_count_eval(lambda db: _setting_int(db, 'daily_report_count'), 30), TIER_SILVER, '📰'), ('stat_chart_hover', tr('achieve.stat_chart_hover.name'), tr('achieve.stat_chart_hover.desc'), 1, _bool_eval(lambda db: db.get_setting('chart_hover_discovered', 'false').lower() == 'true'), TIER_BRONZE, '🎨'), ('stat_achievements_open', tr('achieve.stat_achievements_open.name'), tr('achieve.stat_achievements_open.desc'), 50, _make_count_eval(lambda db: _setting_int(db, 'achievements_view_count'), 50), TIER_BRONZE, '🦄'), ] # ---- 14. 시크릿 ---- def _has_clock_in_palindrome(db) -> bool: """출근 시각이 회문 (HH:MM에서 H1H2:M1M2가 회문).""" with db._conn() as conn: cur = conn.cursor() cur.execute("SELECT clock_in FROM work_records WHERE clock_in IS NOT NULL") for (t,) in cur.fetchall(): if not t: continue digits = t[:5].replace(':', '') if len(digits) == 4 and digits == digits[::-1]: return True return False def _has_clock_in_jackpot(db) -> bool: """출근 시각 모든 자릿수 동일 (11:11, 22:22).""" with db._conn() as conn: cur = conn.cursor() cur.execute("SELECT clock_in FROM work_records WHERE clock_in IS NOT NULL") for (t,) in cur.fetchall(): if not t: continue digits = t[:5].replace(':', '') if len(digits) == 4 and len(set(digits)) == 1: return True return False def _has_friday_13th_clockin(db) -> bool: with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT 1 FROM work_records WHERE strftime('%w', date) = '5' AND CAST(strftime('%d', date) AS INTEGER) = 13 LIMIT 1 """) return cur.fetchone() is not None def _has_777(db) -> bool: """7월 7일 7시 7분 출근.""" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT 1 FROM work_records WHERE strftime('%m-%d', date) = '07-07' AND clock_in LIKE '07:07%' LIMIT 1 """) return cur.fetchone() is not None def _has_exact_8h(db) -> bool: """정확히 8시간 0분 0초 근무.""" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT 1 FROM work_records WHERE clock_in IS NOT NULL AND clock_out IS NOT NULL AND total_hours = 8.0 LIMIT 1 """) return cur.fetchone() is not None def _has_pi_day(db) -> bool: """3/14 1:59 출근 (π = 3.14159).""" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT 1 FROM work_records WHERE strftime('%m-%d', date) = '03-14' AND clock_in LIKE '01:59%' LIMIT 1 """) return cur.fetchone() is not None def _has_fibonacci_minute(db) -> bool: """출근 시각 분이 피보나치 (1,2,3,5,8,13,21,34,55).""" fibs = {1, 2, 3, 5, 8, 13, 21, 34, 55} with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT CAST(strftime('%M', clock_in) AS INTEGER) FROM work_records WHERE clock_in IS NOT NULL """) return any(r[0] in fibs for r in cur.fetchall()) def _has_double_six(db) -> bool: """6/6 18:06 출근.""" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT 1 FROM work_records WHERE strftime('%m-%d', date) = '06-06' AND clock_in LIKE '18:06%' LIMIT 1 """) return cur.fetchone() is not None def _has_500_anniv_clockin(db) -> bool: """가입 후 정확히 365일 후 출근.""" hire = _setting_str(db, 'hire_date', '') if not hire: return False try: d = datetime.strptime(hire, '%Y-%m-%d').date() target = d + timedelta(days=365) return _has_clockin_on_date(db, target) except ValueError: return False _SECRET_DEFS = [ ('secret_palindrome', tr('achieve.secret_palindrome.name'), tr('achieve.secret_palindrome.desc'), 1, _bool_eval(_has_clock_in_palindrome), TIER_GOLD, '🪞'), ('secret_jackpot', tr('achieve.secret_jackpot.name'), tr('achieve.secret_jackpot.desc'), 1, _bool_eval(_has_clock_in_jackpot), TIER_PLATINUM, '🎰'), ('secret_fri13', tr('achieve.secret_fri13.name'), tr('achieve.secret_fri13.desc'), 1, _bool_eval(_has_friday_13th_clockin), TIER_GOLD, '🌑'), ('secret_777', tr('achieve.secret_777.name'), tr('achieve.secret_777.desc'), 1, _bool_eval(_has_777), TIER_LEGEND, '🔮'), ('secret_exact_8h', tr('achieve.secret_exact_8h.name'), tr('achieve.secret_exact_8h.desc'), 1, _bool_eval(_has_exact_8h), TIER_PLATINUM, '🎯'), ('secret_pi_day', tr('achieve.secret_pi_day.name'), tr('achieve.secret_pi_day.desc'), 1, _bool_eval(_has_pi_day), TIER_LEGEND, '🥧'), ('secret_fibonacci', tr('achieve.secret_fibonacci.name'), tr('achieve.secret_fibonacci.desc'), 1, _bool_eval(_has_fibonacci_minute), TIER_SILVER, '🔢'), ('secret_double_six', tr('achieve.secret_double_six.name'), tr('achieve.secret_double_six.desc'), 1, _bool_eval(_has_double_six), TIER_LEGEND, '🎲'), ('secret_anniversary', tr('achieve.secret_anniversary.name'), tr('achieve.secret_anniversary.desc'), 1, _bool_eval(_has_500_anniv_clockin), TIER_LEGEND, '🧙'), ] # ---- 15. 다양성·설정 ---- def _setting_changed_from_default(db, key: str, default_value: str) -> bool: return str(db.get_setting(key, default_value)) != default_value _SETTINGS_DEFS = [ ('set_dark', tr('achieve.set_dark.name'), tr('achieve.set_dark.desc'), 1, _bool_eval(lambda db: _setting_changed_from_default(db, 'theme', 'light')), TIER_BRONZE, '🌗'), ('set_lang', tr('achieve.set_lang.name'), tr('achieve.set_lang.desc'), 1, _bool_eval(lambda db: db.get_setting('language', 'ko') == 'en'), TIER_BRONZE, '🌐'), ('set_a11y', tr('achieve.set_a11y.name'), tr('achieve.set_a11y.desc'), 1, _bool_eval(lambda db: db.get_setting('font_scale', '1.0') != '1.0' or db.get_setting('high_contrast', 'false').lower() == 'true'), TIER_BRONZE, '♿'), ('set_overtime_unit', tr('achieve.set_overtime_unit.name'), tr('achieve.set_overtime_unit.desc'), 1, _bool_eval(lambda db: db.get_setting('overtime_unit', '30') != '30'), TIER_BRONZE, '⏱️'), ('set_goal_full', tr('achieve.set_goal_full.name'), tr('achieve.set_goal_full.desc'), 1, _bool_eval(lambda db: _setting_int(db, 'goal_overtime_max_monthly') > 0 and float(db.get_setting('goal_avg_hours_daily', '0') or 0) > 0), TIER_SILVER, '🎯'), ('set_discord_full', tr('achieve.set_discord_full.name'), tr('achieve.set_discord_full.desc'), 1, _bool_eval(lambda db: bool(db.get_setting('discord_webhook_url', '') or '') and all(db.get_setting(k, 'true').lower() == 'true' for k in ('notification_clock_out', 'notification_lunch', 'notification_overtime', 'notification_health'))), TIER_SILVER, '🔔'), ('set_cloud', tr('achieve.set_cloud.name'), tr('achieve.set_cloud.desc'), 1, _bool_eval(lambda db: bool(db.get_setting('db_path_override', '') or '')), TIER_SILVER, '☁️'), ] # ---- 16. 메타 (도전과제 자체) ---- def _earned_count(db) -> int: with db._conn() as conn: cur = conn.cursor() cur.execute("SELECT COUNT(*) FROM achievements WHERE earned_date IS NOT NULL") return cur.fetchone()[0] def _earned_secret_count(db) -> int: with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT COUNT(*) FROM achievements WHERE earned_date IS NOT NULL AND is_secret = 1 """) return cur.fetchone()[0] _META_DEFS = [ ('meta_first', tr('achieve.meta_first.name'), tr('achieve.meta_first.desc'), 1, _make_count_eval(_earned_count, 1), TIER_BRONZE, '🏆'), ('meta_10', tr('achieve.meta_10.name'), tr('achieve.meta_10.desc'), 10, _make_count_eval(_earned_count, 10), TIER_BRONZE, '🎖️'), ('meta_25', tr('achieve.meta_25.name'), tr('achieve.meta_25.desc'), 25, _make_count_eval(_earned_count, 25), TIER_SILVER, '🥈'), ('meta_50', tr('achieve.meta_50.name'), tr('achieve.meta_50.desc'), 50, _make_count_eval(_earned_count, 50), TIER_GOLD, '🥇'), ('meta_75', tr('achieve.meta_75.name'), tr('achieve.meta_75.desc'), 75, _make_count_eval(_earned_count, 75), TIER_PLATINUM, '💎'), ('meta_100', tr('achieve.meta_100.name'), tr('achieve.meta_100.desc'), 100, _make_count_eval(_earned_count, 100), TIER_LEGEND, '🌟'), ('meta_secret_1', tr('achieve.meta_secret_1.name'), tr('achieve.meta_secret_1.desc'), 1, _make_count_eval(_earned_secret_count, 1), TIER_SILVER, '🔍'), ('meta_secret_5', tr('achieve.meta_secret_5.name'), tr('achieve.meta_secret_5.desc'), 5, _make_count_eval(_earned_secret_count, 5), TIER_GOLD, '🌑'), ] # ============================================================ # 모든 도전과제 통합 # ============================================================ def _build_all() -> List[Achievement]: """모든 카테고리를 합쳐 Achievement 리스트로 반환.""" all_defs = [] sections = [ (CAT_STREAK, _STREAK_DEFS), (CAT_PUNCTUAL, _PUNCTUAL_DEFS), (CAT_BALANCE, _BALANCE_DEFS), (CAT_OT_BANK, _OT_BANK_DEFS), (CAT_OT_USE, _OT_USE_DEFS), (CAT_LEAVE, _LEAVE_DEFS), (CAT_MEAL, _MEAL_DEFS), (CAT_BREAK, _BREAK_DEFS), (CAT_TIME_SLOT, _TIME_SLOT_DEFS), (CAT_SPECIAL_DAY, _SPECIAL_DAY_DEFS), (CAT_SEASON, _SEASON_DEFS), (CAT_MILESTONE, _MILESTONE_DEFS), (CAT_STATS, _STATS_DEFS), (CAT_SETTINGS, _SETTINGS_DEFS), (CAT_SECRET, _SECRET_DEFS), (CAT_META, _META_DEFS), ] for cat, defs in sections: for tup in defs: code, name, desc, target, evaluator, tier, icon = tup is_secret = (cat == CAT_SECRET) all_defs.append(Achievement( code=code, name=name, description=desc, category=cat, tier=tier, badge_icon=icon, is_secret=is_secret, target=target, evaluator=evaluator, )) return all_defs ALL_ACHIEVEMENTS: List[Achievement] = _build_all() def get_achievement(code: str) -> Optional[Achievement]: for a in ALL_ACHIEVEMENTS: if a.code == code: return a return None # ============================================================ # DB 동기화 + 평가 # ============================================================ def sync_definitions_to_db(db) -> None: """ALL_ACHIEVEMENTS 정의를 achievements 테이블에 upsert. code가 unique key. 신규 도전과제는 INSERT, 기존은 메타데이터 UPDATE (earned_date, progress는 보존). """ with db._conn() as conn: cur = conn.cursor() for a in ALL_ACHIEVEMENTS: cur.execute("SELECT id FROM achievements WHERE code = ?", (a.code,)) row = cur.fetchone() if row is None: cur.execute(''' INSERT INTO achievements (code, name, description, category, tier, is_secret, progress, target, badge_icon) VALUES (?, ?, ?, ?, ?, ?, 0, ?, ?) ''', (a.code, a.name, a.description, a.category, a.tier, 1 if a.is_secret else 0, a.target, a.badge_icon)) else: cur.execute(''' UPDATE achievements SET name = ?, description = ?, category = ?, tier = ?, is_secret = ?, target = ?, badge_icon = ? WHERE code = ? ''', (a.name, a.description, a.category, a.tier, 1 if a.is_secret else 0, a.target, a.badge_icon, a.code)) conn.commit() def evaluate_all(db) -> List[Achievement]: """미획득 도전과제만 평가. 새로 잠금 해제된 것 리스트 반환. side effect: progress 업데이트, earned_date 기록. """ newly_unlocked = [] with db._conn() as conn: cur = conn.cursor() for a in ALL_ACHIEVEMENTS: if a.evaluator is None: continue cur.execute(""" SELECT progress, earned_date FROM achievements WHERE code = ? """, (a.code,)) row = cur.fetchone() if row is None: continue stored_progress, earned = row[0], row[1] if earned is not None: continue # 이미 획득 try: progress, target = a.evaluator(db) except Exception: continue # 평가 실패는 silent (다음 tick에 재시도) now_unlocked = progress >= target if progress != stored_progress or now_unlocked: if now_unlocked: cur.execute(""" UPDATE achievements SET progress = ?, earned_date = DATE('now', 'localtime') WHERE code = ? """, (target, a.code)) newly_unlocked.append(a) else: cur.execute(""" UPDATE achievements SET progress = ? WHERE code = ? """, (progress, a.code)) conn.commit() return newly_unlocked def get_all_with_status(db) -> List[dict]: """UI용: 모든 도전과제 + 진행도/획득 상태 dict 리스트.""" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT code, name, description, category, tier, is_secret, progress, target, earned_date, badge_icon FROM achievements ORDER BY CASE WHEN earned_date IS NOT NULL THEN 0 ELSE 1 END, category, tier """) return [dict(zip( ['code', 'name', 'description', 'category', 'tier', 'is_secret', 'progress', 'target', 'earned_date', 'badge_icon'], row )) for row in cur.fetchall()] def get_stats(db) -> dict: """전체 통계 — 획득/총개수/비밀발견 등.""" with db._conn() as conn: cur = conn.cursor() cur.execute(""" SELECT COUNT(*) AS total, SUM(CASE WHEN earned_date IS NOT NULL THEN 1 ELSE 0 END) AS earned, SUM(CASE WHEN is_secret = 1 THEN 1 ELSE 0 END) AS secret_total, SUM(CASE WHEN is_secret = 1 AND earned_date IS NOT NULL THEN 1 ELSE 0 END) AS secret_earned FROM achievements """) row = cur.fetchone() return { 'total': row[0] or 0, 'earned': row[1] or 0, 'secret_total': row[2] or 0, 'secret_earned': row[3] or 0, }