mirror of
https://github.com/ankitects/anki.git
synced 2025-09-18 14:02:21 -04:00

* Update to latest Node LTS * Add sveltekit * Split tslib into separate @generated and @tslib components SvelteKit's path aliases don't support multiple locations, so our old approach of using @tslib to refer to both ts/lib and out/ts/lib will no longer work. Instead, all generated sources and their includes are placed in a separate out/ts/generated folder, and imported via @generated instead. This also allows us to generate .ts files, instead of needing to output separate .d.ts and .js files. * Switch package.json to module type * Avoid usage of baseUrl Incompatible with SvelteKit * Move sass into ts; use relative links SvelteKit's default sass support doesn't allow overriding loadPaths * jest->vitest, graphs example working with yarn dev * most pages working in dev mode * Some fixes after rebasing * Fix/silence some svelte-check errors * Get image-occlusion working with Fabric types * Post-rebase lock changes * Editor is now checked * SvelteKit build integrated into ninja * Use the new SvelteKit entrypoint for pages like congrats/deck options/etc * Run eslint once for ts/**; fix some tests * Fix a bunch of issues introduced when rebasing over latest main * Run eslint fix * Fix remaining eslint+pylint issues; tests now all pass * Fix some issues with a clean build * Latest bufbuild no longer requires @__PURE__ hack * Add a few missed dependencies * Add yarn.bat to fix Windows build * Fix pages failing to show when ANKI_API_PORT not defined * Fix svelte-check and vitest on Windows * Set node path in ./yarn * Move svelte-kit output to ts/.svelte-kit Sadly, I couldn't figure out a way to store it in out/ if out/ is a symlink, as it breaks module resolution when SvelteKit is run. * Allow HMR inside Anki * Skip SvelteKit build when HMR is defined * Fix some post-rebase issues I should have done a normal merge instead.
150 lines
4.3 KiB
Python
150 lines
4.3 KiB
Python
# Copyright: Ankitects Pty Ltd and contributors
|
|
# License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
|
|
|
from __future__ import annotations
|
|
|
|
from typing import Callable
|
|
|
|
import aqt
|
|
from anki.cards import Card, CardId
|
|
from anki.errors import NotFoundError
|
|
from anki.lang import without_unicode_isolation
|
|
from aqt.qt import *
|
|
from aqt.utils import (
|
|
addCloseShortcut,
|
|
disable_help_button,
|
|
qconnect,
|
|
restoreGeom,
|
|
saveGeom,
|
|
setWindowIcon,
|
|
tr,
|
|
)
|
|
from aqt.webview import AnkiWebView, AnkiWebViewKind
|
|
|
|
|
|
class CardInfoDialog(QDialog):
|
|
TITLE = "browser card info"
|
|
GEOMETRY_KEY = "revlog"
|
|
silentlyClose = True
|
|
|
|
def __init__(
|
|
self,
|
|
parent: QWidget | None,
|
|
mw: aqt.AnkiQt,
|
|
card: Card | None,
|
|
on_close: Callable | None = None,
|
|
geometry_key: str | None = None,
|
|
window_title: str | None = None,
|
|
) -> None:
|
|
super().__init__(parent)
|
|
self.mw = mw
|
|
self._on_close = on_close
|
|
self.GEOMETRY_KEY = geometry_key or self.GEOMETRY_KEY
|
|
if window_title:
|
|
self.setWindowTitle(window_title)
|
|
self._setup_ui(card.id if card else None)
|
|
self.show()
|
|
|
|
def _setup_ui(self, card_id: CardId | None) -> None:
|
|
self.mw.garbage_collect_on_dialog_finish(self)
|
|
disable_help_button(self)
|
|
restoreGeom(self, self.GEOMETRY_KEY, default_size=(800, 800))
|
|
addCloseShortcut(self)
|
|
setWindowIcon(self)
|
|
|
|
self.web = AnkiWebView(kind=AnkiWebViewKind.BROWSER_CARD_INFO)
|
|
self.web.setVisible(False)
|
|
self.web.load_sveltekit_page(f"card-info/{card_id}")
|
|
layout = QVBoxLayout()
|
|
layout.setContentsMargins(0, 0, 0, 0)
|
|
layout.addWidget(self.web)
|
|
buttons = QDialogButtonBox(QDialogButtonBox.StandardButton.Close)
|
|
buttons.setContentsMargins(10, 0, 10, 10)
|
|
layout.addWidget(buttons)
|
|
qconnect(buttons.rejected, self.reject)
|
|
self.setLayout(layout)
|
|
|
|
def update_card(self, card_id: CardId | None) -> None:
|
|
try:
|
|
self.mw.col.get_card(card_id)
|
|
except NotFoundError:
|
|
card_id = None
|
|
self.web.eval(f"window.location.href = '/card-info/{card_id}';")
|
|
|
|
def reject(self) -> None:
|
|
if self._on_close:
|
|
self._on_close()
|
|
self.web.cleanup()
|
|
self.web = None
|
|
saveGeom(self, self.GEOMETRY_KEY)
|
|
return QDialog.reject(self)
|
|
|
|
|
|
class CardInfoManager:
|
|
"""Wrapper class to conveniently toggle, update and close a card info dialog."""
|
|
|
|
def __init__(self, mw: aqt.AnkiQt, geometry_key: str, window_title: str):
|
|
self.mw = mw
|
|
self.geometry_key = geometry_key
|
|
self.window_title = window_title
|
|
self._card: Card | None = None
|
|
self._dialog: CardInfoDialog | None = None
|
|
|
|
def show(self) -> None:
|
|
if self._dialog:
|
|
self._dialog.activateWindow()
|
|
self._dialog.raise_()
|
|
else:
|
|
self._dialog = CardInfoDialog(
|
|
None,
|
|
self.mw,
|
|
self._card,
|
|
self._on_close,
|
|
self.geometry_key,
|
|
self.window_title,
|
|
)
|
|
|
|
def set_card(self, card: Card | None) -> None:
|
|
self._card = card
|
|
if self._dialog:
|
|
self._dialog.update_card(card.id if card else None)
|
|
|
|
def close(self) -> None:
|
|
if self._dialog:
|
|
self._dialog.reject()
|
|
|
|
def _on_close(self) -> None:
|
|
self._dialog = None
|
|
|
|
|
|
class BrowserCardInfo(CardInfoManager):
|
|
def __init__(self, mw: aqt.AnkiQt):
|
|
super().__init__(
|
|
mw,
|
|
"revlog",
|
|
without_unicode_isolation(
|
|
tr.card_stats_current_card(context=tr.qt_misc_browse())
|
|
),
|
|
)
|
|
|
|
|
|
class ReviewerCardInfo(CardInfoManager):
|
|
def __init__(self, mw: aqt.AnkiQt):
|
|
super().__init__(
|
|
mw,
|
|
"reviewerCardInfo",
|
|
without_unicode_isolation(
|
|
tr.card_stats_current_card(context=tr.decks_study())
|
|
),
|
|
)
|
|
|
|
|
|
class PreviousReviewerCardInfo(CardInfoManager):
|
|
def __init__(self, mw: aqt.AnkiQt):
|
|
super().__init__(
|
|
mw,
|
|
"previousReviewerCardInfo",
|
|
without_unicode_isolation(
|
|
tr.card_stats_previous_card(context=tr.decks_study())
|
|
),
|
|
)
|