- 数据库 v3: priority → quadrant (重要紧急/重要不紧急/紧急不重要/不紧急不重要) - 2×2 网格布局,每象限独立列表 - 跨象限拖拽:拖到不同象限自动更新分类 - 旧数据自动迁移:高→重要紧急, 中→重要不紧急, 低→不紧急不重要
204 lines
6.4 KiB
Python
204 lines
6.4 KiB
Python
# -*- coding: utf-8 -*-
|
||
"""ziwei-power SQLite 数据库操作层"""
|
||
|
||
import sqlite3
|
||
import json
|
||
import os
|
||
from datetime import datetime
|
||
|
||
DB_DIR = os.path.join(os.path.expanduser('~'), '.workbuddy', 'data', 'ziwei-power')
|
||
os.makedirs(DB_DIR, exist_ok=True)
|
||
DB_PATH = os.path.join(DB_DIR, 'ziwei_power.db')
|
||
|
||
# 当前数据库 schema 版本 —— 改表结构时必须 +1 并补迁移逻辑
|
||
CURRENT_SCHEMA_VERSION = 3
|
||
|
||
|
||
def get_conn():
|
||
conn = sqlite3.connect(DB_PATH)
|
||
conn.row_factory = sqlite3.Row
|
||
return conn
|
||
|
||
|
||
def _get_schema_version(conn):
|
||
"""读取当前数据库的 schema 版本,无表时返回 0"""
|
||
conn.execute('''
|
||
CREATE TABLE IF NOT EXISTS schema_version (
|
||
version INTEGER NOT NULL
|
||
)
|
||
''')
|
||
row = conn.execute('SELECT version FROM schema_version').fetchone()
|
||
return row['version'] if row else 0
|
||
|
||
|
||
def _set_schema_version(conn, version):
|
||
"""写入 schema 版本"""
|
||
conn.execute('DELETE FROM schema_version')
|
||
conn.execute('INSERT INTO schema_version (version) VALUES (?)', (version,))
|
||
|
||
|
||
def init_db():
|
||
"""初始化数据库表 & 自动迁移"""
|
||
conn = get_conn()
|
||
current = _get_schema_version(conn)
|
||
|
||
# ── 迁移步骤(按版本号递增)────────────────────────
|
||
if current < 1:
|
||
# v1: 初始表结构
|
||
conn.execute('''
|
||
CREATE TABLE IF NOT EXISTS checkins (
|
||
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||
date TEXT UNIQUE NOT NULL,
|
||
data TEXT NOT NULL,
|
||
created_at TEXT NOT NULL,
|
||
updated_at TEXT NOT NULL
|
||
)
|
||
''')
|
||
|
||
if current < 2:
|
||
# v2: 心愿清单
|
||
conn.execute('''
|
||
CREATE TABLE IF NOT EXISTS wishes (
|
||
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||
name TEXT NOT NULL,
|
||
priority TEXT NOT NULL DEFAULT '中',
|
||
deadline TEXT NOT NULL DEFAULT '',
|
||
done INTEGER NOT NULL DEFAULT 0,
|
||
sort_order INTEGER NOT NULL DEFAULT 0,
|
||
created_at TEXT NOT NULL
|
||
)
|
||
''')
|
||
|
||
if current < 3:
|
||
# v3: 优先级改为四象限
|
||
conn.execute("ALTER TABLE wishes ADD COLUMN quadrant TEXT NOT NULL DEFAULT '重要不紧急'")
|
||
conn.execute("UPDATE wishes SET quadrant = CASE priority WHEN '高' THEN '重要紧急' WHEN '中' THEN '重要不紧急' WHEN '低' THEN '不紧急不重要' ELSE '重要不紧急' END WHERE quadrant = '重要不紧急'")
|
||
|
||
# ── 将来加字段/改表在此扩展 ──
|
||
# if current < 2:
|
||
# conn.execute('ALTER TABLE checkins ADD COLUMN tags TEXT DEFAULT ""')
|
||
# # 可选:对已有行做数据补全
|
||
# conn.execute("UPDATE checkins SET tags = '[]' WHERE tags IS NULL")
|
||
|
||
# ── 写入最新版本号 ──
|
||
_set_schema_version(conn, CURRENT_SCHEMA_VERSION)
|
||
conn.commit()
|
||
conn.close()
|
||
|
||
|
||
def get_checkin(date_str):
|
||
"""获取某天的打卡记录,返回 dict 或 None"""
|
||
conn = get_conn()
|
||
row = conn.execute('SELECT * FROM checkins WHERE date = ?', (date_str,)).fetchone()
|
||
conn.close()
|
||
if row:
|
||
return {
|
||
'id': row['id'],
|
||
'date': row['date'],
|
||
'data': json.loads(row['data']),
|
||
'created_at': row['created_at'],
|
||
'updated_at': row['updated_at']
|
||
}
|
||
return None
|
||
|
||
|
||
def save_checkin(date_str, data_dict):
|
||
"""保存或更新打卡记录"""
|
||
now = datetime.now().isoformat()
|
||
conn = get_conn()
|
||
existing = conn.execute('SELECT id FROM checkins WHERE date = ?', (date_str,)).fetchone()
|
||
json_data = json.dumps(data_dict, ensure_ascii=False)
|
||
if existing:
|
||
conn.execute(
|
||
'UPDATE checkins SET data = ?, updated_at = ? WHERE date = ?',
|
||
(json_data, now, date_str)
|
||
)
|
||
else:
|
||
conn.execute(
|
||
'INSERT INTO checkins (date, data, created_at, updated_at) VALUES (?, ?, ?, ?)',
|
||
(date_str, json_data, now, now)
|
||
)
|
||
conn.commit()
|
||
conn.close()
|
||
|
||
|
||
def delete_checkin(date_str):
|
||
"""删除某天的打卡记录"""
|
||
conn = get_conn()
|
||
conn.execute('DELETE FROM checkins WHERE date = ?', (date_str,))
|
||
conn.commit()
|
||
conn.close()
|
||
|
||
|
||
def get_all_checkins():
|
||
"""获取所有打卡记录,按日期倒序"""
|
||
conn = get_conn()
|
||
rows = conn.execute('SELECT * FROM checkins ORDER BY date DESC').fetchall()
|
||
conn.close()
|
||
results = []
|
||
for row in rows:
|
||
results.append({
|
||
'id': row['id'],
|
||
'date': row['date'],
|
||
'data': json.loads(row['data']),
|
||
'created_at': row['created_at'],
|
||
'updated_at': row['updated_at']
|
||
})
|
||
return results
|
||
|
||
|
||
# ── 心愿清单 CRUD ──────────────────────────────────
|
||
|
||
def get_wishes():
|
||
"""获取所有心愿,按 sort_order 排序"""
|
||
conn = get_conn()
|
||
rows = conn.execute('SELECT * FROM wishes ORDER BY sort_order').fetchall()
|
||
conn.close()
|
||
return [dict(row) for row in rows]
|
||
|
||
|
||
def save_wish(name, quadrant, deadline):
|
||
"""新增一条心愿"""
|
||
now = datetime.now().isoformat()
|
||
conn = get_conn()
|
||
max_order = conn.execute('SELECT COALESCE(MAX(sort_order), -1) + 1 AS n FROM wishes').fetchone()['n']
|
||
conn.execute(
|
||
'INSERT INTO wishes (name, quadrant, deadline, done, sort_order, created_at) VALUES (?, ?, ?, 0, ?, ?)',
|
||
(name, quadrant, deadline, max_order, now)
|
||
)
|
||
conn.commit()
|
||
wish_id = conn.execute('SELECT last_insert_rowid()').fetchone()[0]
|
||
conn.close()
|
||
return wish_id
|
||
|
||
|
||
def update_wish(wish_id, **kwargs):
|
||
"""更新心愿字段"""
|
||
allowed = ['name', 'quadrant', 'deadline', 'done']
|
||
updates = {k: v for k, v in kwargs.items() if k in allowed}
|
||
if not updates:
|
||
return
|
||
conn = get_conn()
|
||
sets = ', '.join(f'{k} = ?' for k in updates)
|
||
vals = list(updates.values()) + [wish_id]
|
||
conn.execute(f'UPDATE wishes SET {sets} WHERE id = ?', vals)
|
||
conn.commit()
|
||
conn.close()
|
||
|
||
|
||
def delete_wish(wish_id):
|
||
"""删除心愿"""
|
||
conn = get_conn()
|
||
conn.execute('DELETE FROM wishes WHERE id = ?', (wish_id,))
|
||
conn.commit()
|
||
conn.close()
|
||
|
||
|
||
def reorder_wishes(order_list):
|
||
"""批量更新排序:order_list = [id1, id2, ...]"""
|
||
conn = get_conn()
|
||
for idx, wid in enumerate(order_list):
|
||
conn.execute('UPDATE wishes SET sort_order = ? WHERE id = ?', (idx, wid))
|
||
conn.commit()
|
||
conn.close()
|