- Backend: EBookHighlights and EBookBookmarks models with encrypted blob storage; GET/POST views with size guards (700 KB / 100 KB); migration applied - Reader header: search, settings, bookmark add/list buttons - Font & layout settings panel (font size, line height, max width, themes for EPUB; zoom, invert, paginated for PDF); persisted in localStorage - Bookmarks: encrypted per-book blob, toast on add, sidebar with jump/delete - Full-text search: EPUB TreeWalker mark injection, PDF span search; Ctrl+F / F3; arrow key cycling; highlights re-applied on search clear - PDF paginated mode: single-page view, tap-zone / swipe / arrow key navigation, smart zoom (text bounding box → scale+translate canvas), auto-enable on mobile - Progress tracking fixes: save before hiding overlay (was always writing 100%), wait for EPUB images to load before restoring scroll, PDF paginated uses page fraction, sendBeacon cache for unload/visibilitychange reliability - PDF text layer disabled pending overlay rendering fix Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
44 lines
1.9 KiB
Python
44 lines
1.9 KiB
Python
# Generated by Django 6.0.3 on 2026-03-19 11:29
|
|
|
|
import django.db.models.deletion
|
|
from django.conf import settings
|
|
from django.db import migrations, models
|
|
|
|
|
|
class Migration(migrations.Migration):
|
|
|
|
dependencies = [
|
|
('books', '0001_initial'),
|
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
|
]
|
|
|
|
operations = [
|
|
migrations.CreateModel(
|
|
name='EBookBookmarks',
|
|
fields=[
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|
('ct', models.TextField()),
|
|
('iv', models.CharField(max_length=32)),
|
|
('updated_at', models.DateTimeField(auto_now=True)),
|
|
('book', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='bookmarks', to='books.ebook')),
|
|
('user', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='ebook_bookmarks', to=settings.AUTH_USER_MODEL)),
|
|
],
|
|
options={
|
|
'unique_together': {('user', 'book')},
|
|
},
|
|
),
|
|
migrations.CreateModel(
|
|
name='EBookHighlights',
|
|
fields=[
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|
('ct', models.TextField()),
|
|
('iv', models.CharField(max_length=32)),
|
|
('updated_at', models.DateTimeField(auto_now=True)),
|
|
('book', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='highlights', to='books.ebook')),
|
|
('user', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='ebook_highlights', to=settings.AUTH_USER_MODEL)),
|
|
],
|
|
options={
|
|
'unique_together': {('user', 'book')},
|
|
},
|
|
),
|
|
]
|