2019-02-05 04:59:03 +01:00
|
|
|
# Copyright: Ankitects Pty Ltd and contributors
|
2012-12-21 08:51:59 +01:00
|
|
|
# License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
|
|
|
|
2020-12-22 04:01:06 +01:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2017-08-28 10:35:24 +02:00
|
|
|
import io
|
2019-12-20 10:19:03 +01:00
|
|
|
import pickle
|
|
|
|
import random
|
|
|
|
import shutil
|
2020-07-02 02:35:01 +02:00
|
|
|
import traceback
|
2020-12-18 07:52:00 +01:00
|
|
|
from enum import Enum
|
2022-03-28 06:40:31 +02:00
|
|
|
from pathlib import Path
|
2021-10-03 10:59:42 +02:00
|
|
|
from typing import Any
|
2013-11-13 07:29:50 +01:00
|
|
|
|
2016-04-05 03:02:01 +02:00
|
|
|
import anki.lang
|
2019-12-20 10:19:03 +01:00
|
|
|
import aqt.forms
|
2020-01-02 10:43:19 +01:00
|
|
|
import aqt.sound
|
2021-07-11 06:51:25 +02:00
|
|
|
from anki.collection import Collection
|
2019-12-20 10:19:03 +01:00
|
|
|
from anki.db import DB
|
2020-11-18 02:53:33 +01:00
|
|
|
from anki.lang import without_unicode_isolation
|
2021-01-31 06:55:08 +01:00
|
|
|
from anki.sync import SyncAuth
|
2022-01-21 12:43:54 +01:00
|
|
|
from anki.utils import int_time, is_mac, is_win, point_version
|
2019-12-20 10:19:03 +01:00
|
|
|
from aqt import appHelpSite
|
|
|
|
from aqt.qt import *
|
2021-11-24 22:17:41 +01:00
|
|
|
from aqt.theme import Theme
|
2022-03-28 06:40:31 +02:00
|
|
|
from aqt.utils import disable_help_button, send_to_trash, showWarning, tr
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2020-11-17 08:42:43 +01:00
|
|
|
# Profile handling
|
|
|
|
##########################################################################
|
|
|
|
# - Saves in pickles rather than json to easily store Qt window state.
|
|
|
|
# - Saves in sqlite rather than a flat file so the config can't be corrupted
|
|
|
|
|
|
|
|
|
2020-12-22 04:01:06 +01:00
|
|
|
class VideoDriver(Enum):
|
|
|
|
OpenGL = "auto"
|
|
|
|
ANGLE = "angle"
|
|
|
|
Software = "software"
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def default_for_platform() -> VideoDriver:
|
2021-11-25 00:06:16 +01:00
|
|
|
if is_mac:
|
2020-12-22 04:01:06 +01:00
|
|
|
return VideoDriver.OpenGL
|
|
|
|
else:
|
|
|
|
return VideoDriver.Software
|
|
|
|
|
|
|
|
def constrained_to_platform(self) -> VideoDriver:
|
2022-04-03 11:57:30 +02:00
|
|
|
if self == VideoDriver.ANGLE and not VideoDriver.supports_angle():
|
2020-12-22 04:01:06 +01:00
|
|
|
return VideoDriver.Software
|
|
|
|
return self
|
|
|
|
|
|
|
|
def next(self) -> VideoDriver:
|
|
|
|
if self == VideoDriver.Software:
|
|
|
|
return VideoDriver.OpenGL
|
2022-04-03 11:57:30 +02:00
|
|
|
elif self == VideoDriver.OpenGL and VideoDriver.supports_angle():
|
2020-12-22 04:01:06 +01:00
|
|
|
return VideoDriver.ANGLE
|
|
|
|
else:
|
|
|
|
return VideoDriver.Software
|
|
|
|
|
2022-04-03 11:57:30 +02:00
|
|
|
@staticmethod
|
|
|
|
def supports_angle() -> bool:
|
|
|
|
return is_win and qtmajor < 6
|
|
|
|
|
2020-12-22 04:01:06 +01:00
|
|
|
@staticmethod
|
2021-10-03 10:59:42 +02:00
|
|
|
def all_for_platform() -> list[VideoDriver]:
|
2020-12-22 04:01:06 +01:00
|
|
|
all = [VideoDriver.OpenGL]
|
2022-04-03 11:57:30 +02:00
|
|
|
if VideoDriver.supports_angle():
|
2020-12-22 04:01:06 +01:00
|
|
|
all.append(VideoDriver.ANGLE)
|
|
|
|
all.append(VideoDriver.Software)
|
|
|
|
return all
|
|
|
|
|
|
|
|
|
2012-12-21 08:51:59 +01:00
|
|
|
metaConf = dict(
|
|
|
|
ver=0,
|
|
|
|
updates=True,
|
2021-10-25 06:50:13 +02:00
|
|
|
created=int_time(),
|
2022-02-25 06:26:06 +01:00
|
|
|
id=random.randrange(0, 2**63),
|
2012-12-21 08:51:59 +01:00
|
|
|
lastMsg=-1,
|
|
|
|
suppressUpdate=False,
|
|
|
|
firstRun=True,
|
|
|
|
defaultLang=None,
|
|
|
|
)
|
|
|
|
|
2021-10-03 10:59:42 +02:00
|
|
|
profileConf: dict[str, Any] = dict(
|
2012-12-21 08:51:59 +01:00
|
|
|
# profile
|
|
|
|
mainWindowGeom=None,
|
|
|
|
mainWindowState=None,
|
2017-10-05 06:39:47 +02:00
|
|
|
numBackups=50,
|
2021-10-25 06:50:13 +02:00
|
|
|
lastOptimize=int_time(),
|
2012-12-21 08:51:59 +01:00
|
|
|
# editing
|
|
|
|
searchHistory=[],
|
2021-08-02 23:12:00 +02:00
|
|
|
lastTextColor="#00f",
|
|
|
|
lastHighlightColor="#00f",
|
2012-12-21 08:51:59 +01:00
|
|
|
# syncing
|
|
|
|
syncKey=None,
|
|
|
|
syncMedia=True,
|
|
|
|
autoSync=True,
|
2012-12-22 05:18:28 +01:00
|
|
|
# importing
|
|
|
|
allowHTML=False,
|
2013-11-26 10:19:54 +01:00
|
|
|
importMode=1,
|
2021-03-10 09:20:37 +01:00
|
|
|
# these are not used, but Anki 2.1.42 and below
|
|
|
|
# expect these keys to exist
|
2021-08-02 23:12:00 +02:00
|
|
|
lastColour="#00f",
|
2021-03-10 09:20:37 +01:00
|
|
|
stripHTML=True,
|
|
|
|
deleteMedia=False,
|
2012-12-21 08:51:59 +01:00
|
|
|
)
|
|
|
|
|
2019-12-23 01:34:10 +01:00
|
|
|
|
2019-12-19 00:38:36 +01:00
|
|
|
class LoadMetaResult:
|
|
|
|
firstTime: bool
|
|
|
|
loadError: bool
|
|
|
|
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2019-12-23 01:34:10 +01:00
|
|
|
class ProfileManager:
|
2021-10-03 10:59:42 +02:00
|
|
|
def __init__(self, base: str | None = None) -> None: #
|
2020-06-01 17:47:46 +02:00
|
|
|
## Settings which should be forgotten each Anki restart
|
2021-10-03 10:59:42 +02:00
|
|
|
self.session: dict[str, Any] = {}
|
|
|
|
self.name: str | None = None
|
|
|
|
self.db: DB | None = None
|
|
|
|
self.profile: dict | None = None
|
2012-12-21 08:51:59 +01:00
|
|
|
# instantiate base folder
|
2020-07-30 16:56:48 +02:00
|
|
|
self.base: str
|
2017-02-15 04:41:19 +01:00
|
|
|
self._setBaseFolder(base)
|
2018-08-08 15:48:25 +02:00
|
|
|
|
2019-12-19 00:38:36 +01:00
|
|
|
def setupMeta(self) -> LoadMetaResult:
|
2012-12-21 08:51:59 +01:00
|
|
|
# load metadata
|
2019-12-19 00:38:36 +01:00
|
|
|
res = self._loadMeta()
|
|
|
|
self.firstRun = res.firstTime
|
|
|
|
return res
|
2018-08-08 15:48:25 +02:00
|
|
|
|
|
|
|
# profile load on startup
|
2021-02-02 14:30:53 +01:00
|
|
|
def openProfile(self, profile: str) -> None:
|
2012-12-21 08:51:59 +01:00
|
|
|
if profile:
|
2016-05-12 06:19:16 +02:00
|
|
|
if profile not in self.profiles():
|
2020-11-21 03:16:26 +01:00
|
|
|
QMessageBox.critical(
|
2021-03-26 04:48:26 +01:00
|
|
|
None, tr.qt_misc_error(), tr.profiles_profile_does_not_exist()
|
2020-11-21 03:16:26 +01:00
|
|
|
)
|
2016-05-12 06:19:16 +02:00
|
|
|
sys.exit(1)
|
2012-12-21 08:51:59 +01:00
|
|
|
try:
|
|
|
|
self.load(profile)
|
2020-08-31 04:04:14 +02:00
|
|
|
except TypeError as exc:
|
|
|
|
raise Exception("Provided profile does not exist.") from exc
|
2012-12-21 08:51:59 +01:00
|
|
|
|
|
|
|
# Base creation
|
|
|
|
######################################################################
|
|
|
|
|
2020-07-30 16:56:48 +02:00
|
|
|
def ensureBaseExists(self) -> None:
|
2019-12-24 11:33:39 +01:00
|
|
|
self._ensureExists(self.base)
|
2012-12-21 08:51:59 +01:00
|
|
|
|
|
|
|
# Profile load/save
|
|
|
|
######################################################################
|
|
|
|
|
2021-10-03 10:59:42 +02:00
|
|
|
def profiles(self) -> list:
|
|
|
|
def names() -> list:
|
2019-03-06 06:27:13 +01:00
|
|
|
return self.db.list("select name from profiles where name != '_global'")
|
|
|
|
|
|
|
|
n = names()
|
|
|
|
if not n:
|
|
|
|
self._ensureProfile()
|
|
|
|
n = names()
|
|
|
|
|
|
|
|
return n
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2021-02-02 14:30:53 +01:00
|
|
|
def _unpickle(self, data: bytes) -> Any:
|
2017-08-28 10:35:24 +02:00
|
|
|
class Unpickler(pickle.Unpickler):
|
2021-10-07 07:36:50 +02:00
|
|
|
def find_class(self, class_module: str, name: str) -> Any:
|
|
|
|
# handle sip lookup ourselves, mapping to current Qt version
|
|
|
|
if class_module == "sip" or class_module.endswith(".sip"):
|
|
|
|
|
|
|
|
def unpickle_type(module: str, klass: str, args: Any) -> Any:
|
|
|
|
if qtmajor > 5:
|
|
|
|
module = module.replace("Qt5", "Qt6")
|
|
|
|
else:
|
|
|
|
module = module.replace("Qt6", "Qt5")
|
|
|
|
if klass == "QByteArray":
|
|
|
|
if module.startswith("PyQt4"):
|
|
|
|
# can't trust str objects from python 2
|
|
|
|
return QByteArray()
|
|
|
|
else:
|
|
|
|
# return the bytes directly
|
|
|
|
return args[0]
|
|
|
|
elif name == "_unpickle_enum":
|
|
|
|
if qtmajor == 5:
|
|
|
|
return sip._unpickle_enum(module, klass, args) # type: ignore
|
|
|
|
else:
|
|
|
|
# old style enums can't be unpickled
|
|
|
|
return None
|
|
|
|
else:
|
|
|
|
return sip._unpickle_type(module, klass, args) # type: ignore
|
|
|
|
|
|
|
|
return unpickle_type
|
2021-11-06 00:32:23 +01:00
|
|
|
else:
|
|
|
|
return super().find_class(class_module, name)
|
2019-12-23 01:34:10 +01:00
|
|
|
|
2017-08-28 10:35:24 +02:00
|
|
|
up = Unpickler(io.BytesIO(data), errors="ignore")
|
|
|
|
return up.load()
|
|
|
|
|
2021-02-02 14:30:53 +01:00
|
|
|
def _pickle(self, obj: Any) -> bytes:
|
2021-10-03 04:08:49 +02:00
|
|
|
for key, val in obj.items():
|
|
|
|
if isinstance(val, QByteArray):
|
|
|
|
obj[key] = bytes(val) # type: ignore
|
|
|
|
|
|
|
|
return pickle.dumps(obj, protocol=4)
|
2017-08-28 10:35:24 +02:00
|
|
|
|
2021-02-02 14:30:53 +01:00
|
|
|
def load(self, name: str) -> bool:
|
2021-11-25 08:47:50 +01:00
|
|
|
if name == "_global":
|
|
|
|
raise Exception("_global is not a valid name")
|
2019-12-23 01:34:10 +01:00
|
|
|
data = self.db.scalar(
|
|
|
|
"select cast(data as blob) from profiles where name = ?", name
|
|
|
|
)
|
2017-08-28 10:35:24 +02:00
|
|
|
self.name = name
|
|
|
|
try:
|
|
|
|
self.profile = self._unpickle(data)
|
|
|
|
except:
|
2021-11-06 00:32:23 +01:00
|
|
|
print(traceback.format_exc())
|
2018-10-29 12:06:33 +01:00
|
|
|
QMessageBox.warning(
|
2019-12-23 01:34:10 +01:00
|
|
|
None,
|
2021-03-26 04:48:26 +01:00
|
|
|
tr.profiles_profile_corrupt(),
|
|
|
|
tr.profiles_anki_could_not_read_your_profile(),
|
2019-12-23 01:34:10 +01:00
|
|
|
)
|
2017-08-28 10:35:24 +02:00
|
|
|
print("resetting corrupt profile")
|
|
|
|
self.profile = profileConf.copy()
|
|
|
|
self.save()
|
2012-12-21 08:51:59 +01:00
|
|
|
return True
|
|
|
|
|
2020-07-30 16:56:48 +02:00
|
|
|
def save(self) -> None:
|
2012-12-21 08:51:59 +01:00
|
|
|
sql = "update profiles set data = ? where name = ?"
|
2017-08-28 10:35:24 +02:00
|
|
|
self.db.execute(sql, self._pickle(self.profile), self.name)
|
|
|
|
self.db.execute(sql, self._pickle(self.meta), "_global")
|
2012-12-21 08:51:59 +01:00
|
|
|
self.db.commit()
|
|
|
|
|
2021-02-02 14:30:53 +01:00
|
|
|
def create(self, name: str) -> None:
|
2012-12-21 08:51:59 +01:00
|
|
|
prof = profileConf.copy()
|
2019-12-23 01:34:10 +01:00
|
|
|
self.db.execute(
|
|
|
|
"insert or ignore into profiles values (?, ?)", name, self._pickle(prof)
|
|
|
|
)
|
2012-12-21 08:51:59 +01:00
|
|
|
self.db.commit()
|
|
|
|
|
2021-02-02 14:30:53 +01:00
|
|
|
def remove(self, name: str) -> None:
|
2022-03-28 06:40:31 +02:00
|
|
|
path = self.profileFolder(create=False)
|
|
|
|
send_to_trash(Path(path))
|
2016-05-12 06:19:16 +02:00
|
|
|
self.db.execute("delete from profiles where name = ?", name)
|
2012-12-21 08:51:59 +01:00
|
|
|
self.db.commit()
|
|
|
|
|
2020-07-30 16:56:48 +02:00
|
|
|
def trashCollection(self) -> None:
|
2022-03-28 06:40:31 +02:00
|
|
|
path = self.collectionPath()
|
|
|
|
send_to_trash(Path(path))
|
2017-08-16 11:45:39 +02:00
|
|
|
|
2021-02-02 14:30:53 +01:00
|
|
|
def rename(self, name: str) -> None:
|
2012-12-21 08:51:59 +01:00
|
|
|
oldName = self.name
|
|
|
|
oldFolder = self.profileFolder()
|
|
|
|
self.name = name
|
|
|
|
newFolder = self.profileFolder(create=False)
|
|
|
|
if os.path.exists(newFolder):
|
2019-12-23 01:34:10 +01:00
|
|
|
if (oldFolder != newFolder) and (oldFolder.lower() == newFolder.lower()):
|
2014-06-18 00:51:02 +02:00
|
|
|
# OS is telling us the folder exists because it does not take
|
|
|
|
# case into account; use a temporary folder location
|
2019-12-23 01:34:10 +01:00
|
|
|
midFolder = "".join([oldFolder, "-temp"])
|
2014-06-18 00:51:02 +02:00
|
|
|
if not os.path.exists(midFolder):
|
|
|
|
os.rename(oldFolder, midFolder)
|
|
|
|
oldFolder = midFolder
|
|
|
|
else:
|
2021-03-26 05:21:04 +01:00
|
|
|
showWarning(tr.profiles_please_remove_the_folder_and(val=midFolder))
|
2014-06-18 00:51:02 +02:00
|
|
|
self.name = oldName
|
|
|
|
return
|
|
|
|
else:
|
2021-03-26 04:48:26 +01:00
|
|
|
showWarning(tr.profiles_folder_already_exists())
|
2014-06-18 00:51:02 +02:00
|
|
|
self.name = oldName
|
|
|
|
return
|
|
|
|
|
2012-12-21 08:51:59 +01:00
|
|
|
# update name
|
2019-12-23 01:34:10 +01:00
|
|
|
self.db.execute("update profiles set name = ? where name = ?", name, oldName)
|
2012-12-21 08:51:59 +01:00
|
|
|
# rename folder
|
2014-06-28 18:22:07 +02:00
|
|
|
try:
|
|
|
|
os.rename(oldFolder, newFolder)
|
2019-03-04 03:29:55 +01:00
|
|
|
except Exception as e:
|
2014-06-28 18:22:07 +02:00
|
|
|
self.db.rollback()
|
2019-02-06 00:02:56 +01:00
|
|
|
if "WinError 5" in str(e):
|
2021-03-26 04:48:26 +01:00
|
|
|
showWarning(tr.profiles_anki_could_not_rename_your_profile())
|
2014-06-28 18:22:07 +02:00
|
|
|
else:
|
|
|
|
raise
|
|
|
|
except:
|
|
|
|
self.db.rollback()
|
|
|
|
raise
|
|
|
|
else:
|
|
|
|
self.db.commit()
|
2012-12-21 08:51:59 +01:00
|
|
|
|
|
|
|
# Folder handling
|
|
|
|
######################################################################
|
|
|
|
|
2021-02-02 14:30:53 +01:00
|
|
|
def profileFolder(self, create: bool = True) -> str:
|
2012-12-21 08:51:59 +01:00
|
|
|
path = os.path.join(self.base, self.name)
|
|
|
|
if create:
|
|
|
|
self._ensureExists(path)
|
|
|
|
return path
|
|
|
|
|
2020-07-30 16:56:48 +02:00
|
|
|
def addonFolder(self) -> str:
|
2016-05-31 10:51:40 +02:00
|
|
|
return self._ensureExists(os.path.join(self.base, "addons21"))
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2020-07-30 16:56:48 +02:00
|
|
|
def backupFolder(self) -> str:
|
2019-12-23 01:34:10 +01:00
|
|
|
return self._ensureExists(os.path.join(self.profileFolder(), "backups"))
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2020-07-30 16:56:48 +02:00
|
|
|
def collectionPath(self) -> str:
|
2012-12-21 08:51:59 +01:00
|
|
|
return os.path.join(self.profileFolder(), "collection.anki2")
|
|
|
|
|
2020-04-16 01:00:49 +02:00
|
|
|
# Downgrade
|
|
|
|
######################################################################
|
|
|
|
|
2021-10-03 10:59:42 +02:00
|
|
|
def downgrade(self, profiles: list[str]) -> list[str]:
|
2020-04-16 01:53:29 +02:00
|
|
|
"Downgrade all profiles. Return a list of profiles that couldn't be opened."
|
|
|
|
problem_profiles = []
|
2020-04-16 01:00:49 +02:00
|
|
|
for name in profiles:
|
|
|
|
path = os.path.join(self.base, name, "collection.anki2")
|
|
|
|
if not os.path.exists(path):
|
|
|
|
continue
|
|
|
|
with DB(path) as db:
|
|
|
|
if db.scalar("select ver from col") == 11:
|
|
|
|
# nothing to do
|
|
|
|
continue
|
2020-04-16 01:53:29 +02:00
|
|
|
try:
|
|
|
|
c = Collection(path)
|
|
|
|
c.close(save=False, downgrade=True)
|
|
|
|
except Exception as e:
|
|
|
|
print(e)
|
|
|
|
problem_profiles.append(name)
|
|
|
|
return problem_profiles
|
2020-04-16 01:00:49 +02:00
|
|
|
|
2012-12-21 08:51:59 +01:00
|
|
|
# Helpers
|
|
|
|
######################################################################
|
|
|
|
|
2020-07-30 16:56:48 +02:00
|
|
|
def _ensureExists(self, path: str) -> str:
|
2012-12-21 08:51:59 +01:00
|
|
|
if not os.path.exists(path):
|
|
|
|
os.makedirs(path)
|
|
|
|
return path
|
|
|
|
|
2021-10-03 10:59:42 +02:00
|
|
|
def _setBaseFolder(self, cmdlineBase: str | None) -> None:
|
2017-02-15 04:41:19 +01:00
|
|
|
if cmdlineBase:
|
|
|
|
self.base = os.path.abspath(cmdlineBase)
|
|
|
|
elif os.environ.get("ANKI_BASE"):
|
|
|
|
self.base = os.path.abspath(os.environ["ANKI_BASE"])
|
|
|
|
else:
|
|
|
|
self.base = self._defaultBase()
|
|
|
|
self.ensureBaseExists()
|
|
|
|
|
2020-07-30 16:56:48 +02:00
|
|
|
def _defaultBase(self) -> str:
|
2021-11-25 00:06:16 +01:00
|
|
|
if is_win:
|
2018-08-08 15:48:25 +02:00
|
|
|
from aqt.winpaths import get_appdata
|
2019-12-23 01:34:10 +01:00
|
|
|
|
2018-08-08 15:48:25 +02:00
|
|
|
return os.path.join(get_appdata(), "Anki2")
|
2021-11-25 00:06:16 +01:00
|
|
|
elif is_mac:
|
2017-01-13 12:14:04 +01:00
|
|
|
return os.path.expanduser("~/Library/Application Support/Anki2")
|
2012-12-21 08:51:59 +01:00
|
|
|
else:
|
2017-02-22 09:16:19 +01:00
|
|
|
dataDir = os.environ.get(
|
2019-12-23 01:34:10 +01:00
|
|
|
"XDG_DATA_HOME", os.path.expanduser("~/.local/share")
|
|
|
|
)
|
2017-02-22 09:16:19 +01:00
|
|
|
if not os.path.exists(dataDir):
|
|
|
|
os.makedirs(dataDir)
|
|
|
|
return os.path.join(dataDir, "Anki2")
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2021-02-02 14:30:53 +01:00
|
|
|
def _loadMeta(self, retrying: bool = False) -> LoadMetaResult:
|
2019-12-19 00:38:36 +01:00
|
|
|
result = LoadMetaResult()
|
|
|
|
result.firstTime = False
|
|
|
|
result.loadError = retrying
|
|
|
|
|
2017-08-28 10:35:24 +02:00
|
|
|
opath = os.path.join(self.base, "prefs.db")
|
2016-05-31 10:51:40 +02:00
|
|
|
path = os.path.join(self.base, "prefs21.db")
|
2020-01-14 05:59:46 +01:00
|
|
|
if not retrying and os.path.exists(opath) and not os.path.exists(path):
|
2017-08-28 10:35:24 +02:00
|
|
|
shutil.copy(opath, path)
|
|
|
|
|
2019-12-19 00:38:36 +01:00
|
|
|
result.firstTime = not os.path.exists(path)
|
2019-12-23 01:34:10 +01:00
|
|
|
|
2020-07-30 16:56:48 +02:00
|
|
|
def recover() -> None:
|
2013-07-18 11:59:14 +02:00
|
|
|
# if we can't load profile, start with a new one
|
2014-07-07 06:18:00 +02:00
|
|
|
if self.db:
|
|
|
|
try:
|
|
|
|
self.db.close()
|
|
|
|
except:
|
|
|
|
pass
|
2018-03-22 05:10:23 +01:00
|
|
|
for suffix in ("", "-journal"):
|
|
|
|
fpath = path + suffix
|
|
|
|
if os.path.exists(fpath):
|
|
|
|
os.unlink(fpath)
|
2019-12-19 00:38:36 +01:00
|
|
|
|
|
|
|
# open DB file and read data
|
2013-07-18 11:59:14 +02:00
|
|
|
try:
|
2016-05-12 06:19:16 +02:00
|
|
|
self.db = DB(path)
|
2021-11-25 08:47:50 +01:00
|
|
|
if not self.db.scalar("pragma integrity_check") == "ok":
|
|
|
|
raise Exception("corrupt db")
|
2019-12-23 01:34:10 +01:00
|
|
|
self.db.execute(
|
|
|
|
"""
|
2012-12-21 08:51:59 +01:00
|
|
|
create table if not exists profiles
|
2020-12-16 06:31:24 +01:00
|
|
|
(name text primary key, data blob not null);"""
|
2019-12-23 01:34:10 +01:00
|
|
|
)
|
2017-08-28 10:35:24 +02:00
|
|
|
data = self.db.scalar(
|
2019-12-23 01:34:10 +01:00
|
|
|
"select cast(data as blob) from profiles where name = '_global'"
|
|
|
|
)
|
2013-07-18 11:59:14 +02:00
|
|
|
except:
|
2020-07-02 02:35:01 +02:00
|
|
|
traceback.print_stack()
|
2019-12-19 00:38:36 +01:00
|
|
|
if result.loadError:
|
|
|
|
# already failed, prevent infinite loop
|
|
|
|
raise
|
|
|
|
# delete files and try again
|
2013-07-18 11:59:14 +02:00
|
|
|
recover()
|
2019-12-19 00:38:36 +01:00
|
|
|
return self._loadMeta(retrying=True)
|
|
|
|
|
|
|
|
# try to read data
|
|
|
|
if not result.firstTime:
|
2013-02-15 00:59:44 +01:00
|
|
|
try:
|
2017-08-28 10:35:24 +02:00
|
|
|
self.meta = self._unpickle(data)
|
2019-12-19 00:38:36 +01:00
|
|
|
return result
|
2013-02-15 00:59:44 +01:00
|
|
|
except:
|
2020-07-02 02:35:01 +02:00
|
|
|
traceback.print_stack()
|
2017-08-28 10:35:24 +02:00
|
|
|
print("resetting corrupt _global")
|
2019-12-19 00:38:36 +01:00
|
|
|
result.loadError = True
|
2020-01-14 06:18:07 +01:00
|
|
|
result.firstTime = True
|
2019-12-19 00:38:36 +01:00
|
|
|
|
|
|
|
# if new or read failed, create a default global profile
|
2013-02-15 00:59:44 +01:00
|
|
|
self.meta = metaConf.copy()
|
2019-12-23 01:34:10 +01:00
|
|
|
self.db.execute(
|
|
|
|
"insert or replace into profiles values ('_global', ?)",
|
|
|
|
self._pickle(metaConf),
|
|
|
|
)
|
2019-12-19 00:38:36 +01:00
|
|
|
return result
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2020-07-30 16:56:48 +02:00
|
|
|
def _ensureProfile(self) -> None:
|
2012-12-21 08:51:59 +01:00
|
|
|
"Create a new profile if none exists."
|
2021-03-26 04:48:26 +01:00
|
|
|
self.create(tr.profiles_user_1())
|
2019-03-06 06:27:13 +01:00
|
|
|
p = os.path.join(self.base, "README.txt")
|
2020-05-10 02:58:42 +02:00
|
|
|
with open(p, "w", encoding="utf8") as file:
|
|
|
|
file.write(
|
|
|
|
without_unicode_isolation(
|
2021-03-26 05:38:15 +01:00
|
|
|
tr.profiles_folder_readme(
|
2021-06-03 08:48:20 +02:00
|
|
|
link=f"{appHelpSite}files#startup-options",
|
2020-10-12 04:17:02 +02:00
|
|
|
)
|
2020-05-10 02:58:42 +02:00
|
|
|
)
|
2021-10-07 05:31:44 +02:00
|
|
|
+ "\n"
|
2019-12-23 01:34:10 +01:00
|
|
|
)
|
2012-12-21 08:51:59 +01:00
|
|
|
|
|
|
|
# Default language
|
|
|
|
######################################################################
|
|
|
|
# On first run, allow the user to choose the default language
|
|
|
|
|
2020-11-21 03:16:26 +01:00
|
|
|
def setDefaultLang(self, idx: int) -> None:
|
2012-12-21 08:51:59 +01:00
|
|
|
# create dialog
|
|
|
|
class NoCloseDiag(QDialog):
|
2021-02-01 14:28:21 +01:00
|
|
|
def reject(self) -> None:
|
2012-12-21 08:51:59 +01:00
|
|
|
pass
|
2019-12-23 01:34:10 +01:00
|
|
|
|
2012-12-21 08:51:59 +01:00
|
|
|
d = self.langDiag = NoCloseDiag()
|
|
|
|
f = self.langForm = aqt.forms.setlang.Ui_Dialog()
|
2020-11-21 03:16:26 +01:00
|
|
|
f.setupUi(d)
|
2021-01-07 05:24:49 +01:00
|
|
|
disable_help_button(d)
|
2020-05-04 05:23:08 +02:00
|
|
|
qconnect(d.accepted, self._onLangSelected)
|
|
|
|
qconnect(d.rejected, lambda: True)
|
2012-12-21 08:51:59 +01:00
|
|
|
# update list
|
2016-04-05 03:02:01 +02:00
|
|
|
f.lang.addItems([x[0] for x in anki.lang.langs])
|
2012-12-21 08:51:59 +01:00
|
|
|
f.lang.setCurrentRow(idx)
|
2021-10-05 02:01:45 +02:00
|
|
|
d.exec()
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2020-07-30 16:56:48 +02:00
|
|
|
def _onLangSelected(self) -> None:
|
2012-12-21 08:51:59 +01:00
|
|
|
f = self.langForm
|
2016-04-05 03:02:01 +02:00
|
|
|
obj = anki.lang.langs[f.lang.currentRow()]
|
2014-01-14 06:40:45 +01:00
|
|
|
code = obj[1]
|
|
|
|
name = obj[0]
|
|
|
|
r = QMessageBox.question(
|
2021-10-05 05:53:01 +02:00
|
|
|
None, "Anki", tr.profiles_confirm_lang_choice(lang=name), QMessageBox.StandardButton.Yes | QMessageBox.StandardButton.No, QMessageBox.StandardButton.No # type: ignore
|
2019-12-23 01:34:10 +01:00
|
|
|
)
|
2021-10-05 05:53:01 +02:00
|
|
|
if r != QMessageBox.StandardButton.Yes:
|
2020-11-21 03:16:26 +01:00
|
|
|
return self.setDefaultLang(f.lang.currentRow())
|
2016-04-05 03:02:01 +02:00
|
|
|
self.setLang(code)
|
|
|
|
|
2021-02-02 14:30:53 +01:00
|
|
|
def setLang(self, code: str) -> None:
|
2019-12-23 01:34:10 +01:00
|
|
|
self.meta["defaultLang"] = code
|
2012-12-21 08:51:59 +01:00
|
|
|
sql = "update profiles set data = ? where name = ?"
|
2017-08-28 10:35:24 +02:00
|
|
|
self.db.execute(sql, self._pickle(self.meta), "_global")
|
2012-12-21 08:51:59 +01:00
|
|
|
self.db.commit()
|
2021-03-26 00:40:41 +01:00
|
|
|
anki.lang.set_lang(code)
|
2018-08-08 15:48:25 +02:00
|
|
|
|
|
|
|
# OpenGL
|
|
|
|
######################################################################
|
|
|
|
|
2020-12-22 04:01:06 +01:00
|
|
|
def _gldriver_path(self) -> str:
|
2018-08-08 15:48:25 +02:00
|
|
|
return os.path.join(self.base, "gldriver")
|
|
|
|
|
2020-12-22 04:01:06 +01:00
|
|
|
def video_driver(self) -> VideoDriver:
|
|
|
|
path = self._gldriver_path()
|
|
|
|
try:
|
2021-10-02 15:20:27 +02:00
|
|
|
with open(path, encoding="utf8") as file:
|
2020-12-22 04:01:06 +01:00
|
|
|
text = file.read().strip()
|
|
|
|
return VideoDriver(text).constrained_to_platform()
|
|
|
|
except (ValueError, OSError):
|
|
|
|
return VideoDriver.default_for_platform()
|
|
|
|
|
|
|
|
def set_video_driver(self, driver: VideoDriver) -> None:
|
2021-10-02 15:20:27 +02:00
|
|
|
with open(self._gldriver_path(), "w", encoding="utf8") as file:
|
2020-12-22 04:01:06 +01:00
|
|
|
file.write(driver.value)
|
|
|
|
|
|
|
|
def set_next_video_driver(self) -> None:
|
|
|
|
self.set_video_driver(self.video_driver().next())
|
2019-12-19 00:58:16 +01:00
|
|
|
|
2020-02-02 23:32:07 +01:00
|
|
|
# Shared options
|
2019-12-19 00:58:16 +01:00
|
|
|
######################################################################
|
|
|
|
|
2022-01-21 12:43:54 +01:00
|
|
|
def last_run_version(self) -> int:
|
|
|
|
return self.meta.get("last_run_version", 0)
|
|
|
|
|
|
|
|
def set_last_run_version(self) -> None:
|
|
|
|
self.meta["last_run_version"] = point_version()
|
|
|
|
|
2019-12-19 00:58:16 +01:00
|
|
|
def uiScale(self) -> float:
|
2020-02-25 07:20:14 +01:00
|
|
|
scale = self.meta.get("uiScale", 1.0)
|
|
|
|
return max(scale, 1)
|
2019-12-19 00:58:16 +01:00
|
|
|
|
|
|
|
def setUiScale(self, scale: float) -> None:
|
|
|
|
self.meta["uiScale"] = scale
|
2020-01-19 02:31:09 +01:00
|
|
|
|
|
|
|
def last_addon_update_check(self) -> int:
|
|
|
|
return self.meta.get("last_addon_update_check", 0)
|
|
|
|
|
2021-02-02 14:30:53 +01:00
|
|
|
def set_last_addon_update_check(self, secs: int) -> None:
|
2020-01-19 02:31:09 +01:00
|
|
|
self.meta["last_addon_update_check"] = secs
|
2020-01-23 06:08:10 +01:00
|
|
|
|
|
|
|
def night_mode(self) -> bool:
|
|
|
|
return self.meta.get("night_mode", False)
|
|
|
|
|
|
|
|
def set_night_mode(self, on: bool) -> None:
|
|
|
|
self.meta["night_mode"] = on
|
2020-02-02 23:32:07 +01:00
|
|
|
|
2021-11-24 22:17:41 +01:00
|
|
|
def theme(self) -> Theme:
|
|
|
|
return Theme(self.meta.get("theme", 0))
|
|
|
|
|
|
|
|
def set_theme(self, theme: Theme) -> None:
|
|
|
|
self.meta["theme"] = theme.value
|
|
|
|
|
2020-04-15 13:37:16 +02:00
|
|
|
def dark_mode_widgets(self) -> bool:
|
|
|
|
return self.meta.get("dark_mode_widgets", False)
|
|
|
|
|
2020-02-04 00:07:15 +01:00
|
|
|
# Profile-specific
|
2020-02-02 23:32:07 +01:00
|
|
|
######################################################################
|
|
|
|
|
2021-10-03 10:59:42 +02:00
|
|
|
def set_sync_key(self, val: str | None) -> None:
|
2020-02-04 03:26:10 +01:00
|
|
|
self.profile["syncKey"] = val
|
|
|
|
|
2021-10-03 10:59:42 +02:00
|
|
|
def set_sync_username(self, val: str | None) -> None:
|
2020-05-30 04:28:22 +02:00
|
|
|
self.profile["syncUser"] = val
|
|
|
|
|
2021-10-03 10:59:42 +02:00
|
|
|
def set_host_number(self, val: int | None) -> None:
|
2020-05-30 04:28:22 +02:00
|
|
|
self.profile["hostNum"] = val or 0
|
|
|
|
|
2020-02-04 03:26:10 +01:00
|
|
|
def media_syncing_enabled(self) -> bool:
|
|
|
|
return self.profile["syncMedia"]
|
|
|
|
|
2020-05-31 02:53:54 +02:00
|
|
|
def auto_syncing_enabled(self) -> bool:
|
|
|
|
return self.profile["autoSync"]
|
|
|
|
|
2021-10-03 10:59:42 +02:00
|
|
|
def sync_auth(self) -> SyncAuth | None:
|
2020-05-30 04:28:22 +02:00
|
|
|
hkey = self.profile.get("syncKey")
|
|
|
|
if not hkey:
|
|
|
|
return None
|
|
|
|
return SyncAuth(hkey=hkey, host_number=self.profile.get("hostNum", 0))
|
2020-02-04 03:46:57 +01:00
|
|
|
|
2020-05-31 03:46:40 +02:00
|
|
|
def clear_sync_auth(self) -> None:
|
|
|
|
self.profile["syncKey"] = None
|
|
|
|
self.profile["syncUser"] = None
|
|
|
|
self.profile["hostNum"] = 0
|
|
|
|
|
2020-07-01 03:35:24 +02:00
|
|
|
def auto_sync_media_minutes(self) -> int:
|
|
|
|
return self.profile.get("autoSyncMediaMinutes", 15)
|
|
|
|
|
2020-07-30 16:56:48 +02:00
|
|
|
def set_auto_sync_media_minutes(self, val: int) -> None:
|
2020-07-01 03:35:24 +02:00
|
|
|
self.profile["autoSyncMediaMinutes"] = val
|
|
|
|
|
2021-09-21 07:03:37 +02:00
|
|
|
def show_browser_table_tooltips(self) -> bool:
|
|
|
|
return self.profile.get("browserTableTooltips", True)
|
|
|
|
|
|
|
|
def set_show_browser_table_tooltips(self, val: bool) -> None:
|
|
|
|
self.profile["browserTableTooltips"] = val
|