mirror of
https://github.com/ankitects/anki.git
synced 2025-11-20 03:27:13 -05:00
- removed 'created' column from various tables. We don't care when things like models are created, and card creation time didn't reflect the actual time a card was created - facts were previously ordered by their creation date. The code would manually set the creation time for subsequent facts on import by 0.0001 seconds, and then card due times were set by adding the fact time to the ordinal number*0.000001. This was prone to error, and the number of zeros used was actually different in different parts of the code. Instead of this, we replace it with a 'pos' column on facts, which increments for each new fact. - importing should add new facts with a higher pos, but concurrent updates in a synced deck can have multiple facts with the same pos - due times are completely different now, and depend on the card type - new cards have due=fact.pos or random(0, 10000) - reviews have due set to an integer representing days since deck creation/download - cards in the learn queue use an integer timestamp in seconds - many columns like modified, lastSync, factor, interval, etc have been converted to integer columns. They are cheaper to store (large decks can save 10s of megabytes) and faster to search for. - cards have their group assigned on fact creation. In the future we'll add a per-template option for a default group. - switch to due/random order for the review queue on upgrade. Users can still switch to the old behaviour if they want, but many people don't care what it's set to, and due is considerably faster, which may result in a better user experience
150 lines
5.1 KiB
Python
150 lines
5.1 KiB
Python
# -*- coding: utf-8 -*-
|
|
# Copyright: Damien Elmes <anki@ichi2.net>
|
|
# License: GNU GPL, version 3 or later; http://www.gnu.org/copyleft/gpl.html
|
|
|
|
import time
|
|
from anki.db import *
|
|
from anki.errors import *
|
|
from anki.models import Model, FieldModel, fieldModelsTable
|
|
from anki.utils import genID, stripHTMLMedia, fieldChecksum, intTime
|
|
from anki.hooks import runHook
|
|
|
|
# Fields in a fact
|
|
##########################################################################
|
|
|
|
fieldsTable = Table(
|
|
'fields', metadata,
|
|
Column('id', Integer, primary_key=True),
|
|
Column('factId', Integer, ForeignKey("facts.id"), nullable=False),
|
|
Column('fieldModelId', Integer, ForeignKey("fieldModels.id"),
|
|
nullable=False),
|
|
Column('ordinal', Integer, nullable=False),
|
|
Column('value', UnicodeText, nullable=False),
|
|
Column('chksum', String, nullable=False, default=""))
|
|
|
|
class Field(object):
|
|
"A field in a fact."
|
|
|
|
def __init__(self, fieldModel=None):
|
|
if fieldModel:
|
|
self.fieldModel = fieldModel
|
|
self.ordinal = fieldModel.ordinal
|
|
self.value = u""
|
|
self.id = genID()
|
|
|
|
def getName(self):
|
|
return self.fieldModel.name
|
|
name = property(getName)
|
|
|
|
mapper(Field, fieldsTable, properties={
|
|
'fieldModel': relation(FieldModel)
|
|
})
|
|
|
|
# Facts: a set of fields and a model
|
|
##########################################################################
|
|
|
|
# Pos: incrementing number defining add order. There may be duplicates if
|
|
# content is added on two sync locations at once. Importing adds to end.
|
|
# Cache: a HTML-stripped amalgam of the field contents, so we can perform
|
|
# searches of marked up text in a reasonable time.
|
|
|
|
factsTable = Table(
|
|
'facts', metadata,
|
|
Column('id', Integer, primary_key=True),
|
|
Column('modelId', Integer, ForeignKey("models.id"), nullable=False),
|
|
Column('pos', Integer, nullable=False),
|
|
Column('modified', Integer, nullable=False, default=intTime),
|
|
Column('tags', UnicodeText, nullable=False, default=u""),
|
|
Column('cache', UnicodeText, nullable=False, default=u""))
|
|
|
|
class Fact(object):
|
|
"A single fact. Fields exposed as dict interface."
|
|
|
|
def __init__(self, model=None, pos=None):
|
|
self.model = model
|
|
self.id = genID()
|
|
if model:
|
|
for fm in model.fieldModels:
|
|
self.fields.append(Field(fm))
|
|
self.pos = pos
|
|
self.new = True
|
|
|
|
def isNew(self):
|
|
return getattr(self, 'new', False)
|
|
|
|
def keys(self):
|
|
return [field.name for field in self.fields]
|
|
|
|
def values(self):
|
|
return [field.value for field in self.fields]
|
|
|
|
def __getitem__(self, key):
|
|
try:
|
|
return [f.value for f in self.fields if f.name == key][0]
|
|
except IndexError:
|
|
raise KeyError(key)
|
|
|
|
def __setitem__(self, key, value):
|
|
try:
|
|
item = [f for f in self.fields if f.name == key][0]
|
|
except IndexError:
|
|
raise KeyError
|
|
item.value = value
|
|
if item.fieldModel.unique:
|
|
item.chksum = fieldChecksum(value)
|
|
else:
|
|
item.chksum = ""
|
|
|
|
def get(self, key, default):
|
|
try:
|
|
return self[key]
|
|
except (IndexError, KeyError):
|
|
return default
|
|
|
|
def assertValid(self):
|
|
"Raise an error if required fields are empty."
|
|
for field in self.fields:
|
|
if not self.fieldValid(field):
|
|
raise FactInvalidError(type="fieldEmpty",
|
|
field=field.name)
|
|
|
|
def fieldValid(self, field):
|
|
return not (field.fieldModel.required and not field.value.strip())
|
|
|
|
def assertUnique(self, s):
|
|
"Raise an error if duplicate fields are found."
|
|
for field in self.fields:
|
|
if not self.fieldUnique(field, s):
|
|
raise FactInvalidError(type="fieldNotUnique",
|
|
field=field.name)
|
|
|
|
def fieldUnique(self, field, s):
|
|
if not field.fieldModel.unique:
|
|
return True
|
|
req = ("select value from fields "
|
|
"where fieldModelId = :fmid and value = :val and chksum = :chk")
|
|
if field.id:
|
|
req += " and id != %s" % field.id
|
|
return not s.scalar(req, val=field.value, fmid=field.fieldModel.id,
|
|
chk=fieldChecksum(field.value))
|
|
|
|
def focusLost(self, field):
|
|
runHook('fact.focusLost', self, field)
|
|
|
|
def setModified(self, textChanged=False, deck=None, media=True):
|
|
"Mark modified and update cards."
|
|
self.modified = intTime()
|
|
if textChanged:
|
|
if not deck:
|
|
# FIXME: compat code
|
|
import ankiqt
|
|
if not getattr(ankiqt, 'setModWarningShown', None):
|
|
import sys; sys.stderr.write(
|
|
"plugin needs to pass deck to fact.setModified()")
|
|
ankiqt.setModWarningShown = True
|
|
deck = ankiqt.mw.deck
|
|
assert deck
|
|
self.cache = stripHTMLMedia(u" ".join(
|
|
self.values()))
|
|
for card in self.cards:
|
|
card.rebuildQA(deck)
|