2020-10-10 03:42:49 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# Copyright: Ankitects Pty Ltd and contributors
|
|
|
|
# License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
2021-01-23 10:59:12 +01:00
|
|
|
|
2020-10-10 03:42:49 +02:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2021-01-04 05:13:20 +01:00
|
|
|
from concurrent.futures import Future
|
2020-10-10 03:42:49 +02:00
|
|
|
from enum import Enum
|
2021-01-29 13:50:29 +01:00
|
|
|
from typing import TYPE_CHECKING, Dict, Iterable, List, Optional, Sequence, Tuple
|
2020-10-10 03:42:49 +02:00
|
|
|
|
|
|
|
import aqt
|
2021-01-29 13:10:31 +01:00
|
|
|
from anki.collection import ConfigBoolKey
|
2020-10-10 03:42:49 +02:00
|
|
|
from anki.errors import DeckRenameError
|
2021-01-29 14:05:30 +01:00
|
|
|
from anki.rsbackend import (
|
|
|
|
DeckTreeNode,
|
|
|
|
FilterToSearchIn,
|
|
|
|
InvalidInput,
|
|
|
|
NamedFilter,
|
|
|
|
TagTreeNode,
|
|
|
|
)
|
2021-01-23 10:59:12 +01:00
|
|
|
from aqt import gui_hooks
|
2021-01-03 08:36:54 +01:00
|
|
|
from aqt.main import ResetReason
|
2021-01-20 00:00:53 +01:00
|
|
|
from aqt.models import Models
|
2020-10-10 03:42:49 +02:00
|
|
|
from aqt.qt import *
|
2021-01-23 10:59:12 +01:00
|
|
|
from aqt.theme import theme_manager
|
2021-01-29 14:05:30 +01:00
|
|
|
from aqt.utils import (
|
|
|
|
TR,
|
|
|
|
askUser,
|
|
|
|
getOnlyText,
|
|
|
|
show_invalid_search_error,
|
|
|
|
showInfo,
|
|
|
|
showWarning,
|
|
|
|
tr,
|
|
|
|
)
|
2020-10-10 03:42:49 +02:00
|
|
|
|
2021-01-29 13:10:31 +01:00
|
|
|
if TYPE_CHECKING:
|
|
|
|
from anki.collection import ConfigBoolKeyValue, TRValue
|
|
|
|
|
2020-10-10 03:42:49 +02:00
|
|
|
|
|
|
|
class SidebarItemType(Enum):
|
|
|
|
ROOT = 0
|
|
|
|
COLLECTION = 1
|
|
|
|
CURRENT_DECK = 2
|
2021-01-29 13:10:31 +01:00
|
|
|
SAVED_SEARCH = 3
|
2020-10-10 03:42:49 +02:00
|
|
|
DECK = 4
|
|
|
|
NOTETYPE = 5
|
|
|
|
TAG = 6
|
|
|
|
CUSTOM = 7
|
2021-01-22 15:49:20 +01:00
|
|
|
TEMPLATE = 8
|
2021-01-29 13:10:31 +01:00
|
|
|
SAVED_SEARCH_ROOT = 9
|
2020-10-10 03:42:49 +02:00
|
|
|
|
|
|
|
|
2021-01-23 10:59:12 +01:00
|
|
|
# used by an add-on hook
|
|
|
|
class SidebarStage(Enum):
|
|
|
|
ROOT = 0
|
|
|
|
STANDARD = 1
|
|
|
|
FAVORITES = 2
|
|
|
|
DECKS = 3
|
|
|
|
MODELS = 4
|
|
|
|
TAGS = 5
|
|
|
|
|
|
|
|
|
|
|
|
class SidebarItem:
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
name: str,
|
|
|
|
icon: str,
|
|
|
|
onClick: Callable[[], None] = None,
|
|
|
|
onExpanded: Callable[[bool], None] = None,
|
|
|
|
expanded: bool = False,
|
|
|
|
item_type: SidebarItemType = SidebarItemType.CUSTOM,
|
|
|
|
id: int = 0,
|
|
|
|
full_name: str = None,
|
|
|
|
) -> None:
|
|
|
|
self.name = name
|
|
|
|
if not full_name:
|
|
|
|
full_name = name
|
|
|
|
self.full_name = full_name
|
|
|
|
self.icon = icon
|
|
|
|
self.item_type = item_type
|
|
|
|
self.id = id
|
|
|
|
self.onClick = onClick
|
|
|
|
self.onExpanded = onExpanded
|
|
|
|
self.expanded = expanded
|
|
|
|
self.children: List["SidebarItem"] = []
|
|
|
|
self.parentItem: Optional["SidebarItem"] = None
|
|
|
|
self.tooltip: Optional[str] = None
|
2021-01-28 09:51:18 +01:00
|
|
|
self.row_in_parent: Optional[int] = None
|
2021-01-23 10:59:12 +01:00
|
|
|
|
|
|
|
def addChild(self, cb: "SidebarItem") -> None:
|
|
|
|
self.children.append(cb)
|
|
|
|
cb.parentItem = self
|
|
|
|
|
|
|
|
def rowForChild(self, child: "SidebarItem") -> Optional[int]:
|
|
|
|
try:
|
|
|
|
return self.children.index(child)
|
|
|
|
except ValueError:
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
class SidebarModel(QAbstractItemModel):
|
|
|
|
def __init__(self, root: SidebarItem) -> None:
|
|
|
|
super().__init__()
|
|
|
|
self.root = root
|
2021-01-28 09:51:18 +01:00
|
|
|
self._cache_rows(root)
|
|
|
|
|
|
|
|
def _cache_rows(self, node: SidebarItem):
|
|
|
|
"Cache index of children in parent."
|
|
|
|
for row, item in enumerate(node.children):
|
|
|
|
item.row_in_parent = row
|
|
|
|
self._cache_rows(item)
|
2021-01-23 10:59:12 +01:00
|
|
|
|
2021-01-28 10:10:06 +01:00
|
|
|
def item_for_index(self, idx: QModelIndex) -> SidebarItem:
|
|
|
|
return idx.internalPointer()
|
|
|
|
|
2021-01-23 10:59:12 +01:00
|
|
|
# Qt API
|
|
|
|
######################################################################
|
|
|
|
|
|
|
|
def rowCount(self, parent: QModelIndex = QModelIndex()) -> int:
|
|
|
|
if not parent.isValid():
|
|
|
|
return len(self.root.children)
|
|
|
|
else:
|
|
|
|
item: SidebarItem = parent.internalPointer()
|
|
|
|
return len(item.children)
|
|
|
|
|
|
|
|
def columnCount(self, parent: QModelIndex = QModelIndex()) -> int:
|
|
|
|
return 1
|
|
|
|
|
|
|
|
def index(
|
|
|
|
self, row: int, column: int, parent: QModelIndex = QModelIndex()
|
|
|
|
) -> QModelIndex:
|
|
|
|
if not self.hasIndex(row, column, parent):
|
|
|
|
return QModelIndex()
|
|
|
|
|
|
|
|
parentItem: SidebarItem
|
|
|
|
if not parent.isValid():
|
|
|
|
parentItem = self.root
|
|
|
|
else:
|
|
|
|
parentItem = parent.internalPointer()
|
|
|
|
|
|
|
|
item = parentItem.children[row]
|
|
|
|
return self.createIndex(row, column, item)
|
|
|
|
|
|
|
|
def parent(self, child: QModelIndex) -> QModelIndex: # type: ignore
|
|
|
|
if not child.isValid():
|
|
|
|
return QModelIndex()
|
|
|
|
|
|
|
|
childItem: SidebarItem = child.internalPointer()
|
|
|
|
parentItem = childItem.parentItem
|
|
|
|
|
|
|
|
if parentItem is None or parentItem == self.root:
|
|
|
|
return QModelIndex()
|
|
|
|
|
2021-01-28 09:51:18 +01:00
|
|
|
row = parentItem.row_in_parent
|
2021-01-23 10:59:12 +01:00
|
|
|
|
|
|
|
return self.createIndex(row, 0, parentItem)
|
|
|
|
|
|
|
|
def data(self, index: QModelIndex, role: int = Qt.DisplayRole) -> QVariant:
|
|
|
|
if not index.isValid():
|
|
|
|
return QVariant()
|
|
|
|
|
2021-01-29 04:18:28 +01:00
|
|
|
if role not in (Qt.DisplayRole, Qt.DecorationRole, Qt.ToolTipRole):
|
2021-01-23 10:59:12 +01:00
|
|
|
return QVariant()
|
|
|
|
|
|
|
|
item: SidebarItem = index.internalPointer()
|
|
|
|
|
|
|
|
if role == Qt.DisplayRole:
|
|
|
|
return QVariant(item.name)
|
|
|
|
elif role == Qt.ToolTipRole:
|
|
|
|
return QVariant(item.tooltip)
|
|
|
|
else:
|
|
|
|
return QVariant(theme_manager.icon_from_resources(item.icon))
|
|
|
|
|
|
|
|
# Helpers
|
|
|
|
######################################################################
|
|
|
|
|
|
|
|
def iconFromRef(self, iconRef: str) -> QIcon:
|
|
|
|
print("iconFromRef() deprecated")
|
|
|
|
return theme_manager.icon_from_resources(iconRef)
|
|
|
|
|
2021-01-26 23:35:40 +01:00
|
|
|
|
2021-01-28 10:10:06 +01:00
|
|
|
def expand_where_necessary(model: SidebarModel, tree: QTreeView, parent=None) -> None:
|
2021-01-28 09:51:18 +01:00
|
|
|
parent = parent or QModelIndex()
|
|
|
|
for row in range(model.rowCount(parent)):
|
|
|
|
idx = model.index(row, 0, parent)
|
|
|
|
if not idx.isValid():
|
|
|
|
continue
|
|
|
|
expand_where_necessary(model, tree, idx)
|
2021-01-28 10:10:06 +01:00
|
|
|
item = model.item_for_index(idx)
|
|
|
|
if item and item.expanded:
|
2021-01-28 09:51:18 +01:00
|
|
|
tree.setExpanded(idx, True)
|
2021-01-26 23:35:40 +01:00
|
|
|
|
|
|
|
|
2021-01-28 10:10:06 +01:00
|
|
|
class FilterModel(QSortFilterProxyModel):
|
|
|
|
def item_for_index(self, idx: QModelIndex) -> Optional[SidebarItem]:
|
|
|
|
if not idx.isValid():
|
|
|
|
return None
|
|
|
|
return self.mapToSource(idx).internalPointer()
|
|
|
|
|
|
|
|
|
2021-01-26 23:35:40 +01:00
|
|
|
class SidebarSearchBar(QLineEdit):
|
2021-01-28 09:51:18 +01:00
|
|
|
def __init__(self, sidebar: SidebarTreeView):
|
2021-01-26 23:35:40 +01:00
|
|
|
QLineEdit.__init__(self, sidebar)
|
2021-01-29 13:10:31 +01:00
|
|
|
self.setPlaceholderText(sidebar.col.tr(TR.BROWSING_SIDEBAR_FILTER))
|
2021-01-26 23:35:40 +01:00
|
|
|
self.sidebar = sidebar
|
2021-01-29 01:40:32 +01:00
|
|
|
self.timer = QTimer(self)
|
|
|
|
self.timer.setInterval(600)
|
|
|
|
self.timer.setSingleShot(True)
|
|
|
|
qconnect(self.timer.timeout, self.onSearch)
|
2021-01-26 23:35:40 +01:00
|
|
|
qconnect(self.textChanged, self.onTextChanged)
|
|
|
|
|
|
|
|
def onTextChanged(self, text: str):
|
2021-01-29 01:40:32 +01:00
|
|
|
if not self.timer.isActive():
|
|
|
|
self.timer.start()
|
|
|
|
|
|
|
|
def onSearch(self):
|
|
|
|
self.sidebar.search_for(self.text())
|
2021-01-26 23:35:40 +01:00
|
|
|
|
|
|
|
def keyPressEvent(self, evt):
|
|
|
|
if evt.key() in (Qt.Key_Up, Qt.Key_Down):
|
|
|
|
self.sidebar.setFocus()
|
|
|
|
elif evt.key() in (Qt.Key_Enter, Qt.Key_Return):
|
2021-01-29 01:40:32 +01:00
|
|
|
self.onSearch()
|
2021-01-26 23:35:40 +01:00
|
|
|
else:
|
|
|
|
QLineEdit.keyPressEvent(self, evt)
|
|
|
|
|
2021-01-23 10:59:12 +01:00
|
|
|
|
|
|
|
class SidebarTreeView(QTreeView):
|
|
|
|
def __init__(self, browser: aqt.browser.Browser) -> None:
|
2020-10-10 03:42:49 +02:00
|
|
|
super().__init__()
|
2021-01-23 10:59:12 +01:00
|
|
|
self.browser = browser
|
|
|
|
self.mw = browser.mw
|
|
|
|
self.col = self.mw.col
|
2021-01-28 10:10:06 +01:00
|
|
|
self.current_search: Optional[str] = None
|
2021-01-23 10:59:12 +01:00
|
|
|
|
|
|
|
self.setContextMenuPolicy(Qt.CustomContextMenu)
|
|
|
|
self.customContextMenuRequested.connect(self.onContextMenu) # type: ignore
|
2021-01-29 13:50:29 +01:00
|
|
|
self.context_menus: Dict[SidebarItemType, Sequence[Tuple[str, Callable]]] = {
|
2021-01-23 10:59:12 +01:00
|
|
|
SidebarItemType.DECK: (
|
|
|
|
(tr(TR.ACTIONS_RENAME), self.rename_deck),
|
|
|
|
(tr(TR.ACTIONS_DELETE), self.delete_deck),
|
|
|
|
),
|
|
|
|
SidebarItemType.TAG: (
|
|
|
|
(tr(TR.ACTIONS_RENAME), self.rename_tag),
|
|
|
|
(tr(TR.ACTIONS_DELETE), self.remove_tag),
|
|
|
|
),
|
2021-01-29 13:10:31 +01:00
|
|
|
SidebarItemType.SAVED_SEARCH: (
|
2021-01-29 14:05:30 +01:00
|
|
|
(tr(TR.ACTIONS_RENAME), self.rename_saved_search),
|
|
|
|
(tr(TR.ACTIONS_DELETE), self.remove_saved_search),
|
2021-01-23 10:59:12 +01:00
|
|
|
),
|
|
|
|
SidebarItemType.NOTETYPE: ((tr(TR.ACTIONS_MANAGE), self.manage_notetype),),
|
2021-01-29 13:10:31 +01:00
|
|
|
SidebarItemType.SAVED_SEARCH_ROOT: (
|
|
|
|
(tr(TR.BROWSING_SIDEBAR_SAVE_CURRENT_SEARCH), self.save_current_search),
|
|
|
|
),
|
2021-01-23 10:59:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
self.setUniformRowHeights(True)
|
|
|
|
self.setHeaderHidden(True)
|
|
|
|
self.setIndentation(15)
|
|
|
|
|
2020-10-10 03:42:49 +02:00
|
|
|
qconnect(self.expanded, self.onExpansion)
|
|
|
|
qconnect(self.collapsed, self.onCollapse)
|
|
|
|
|
2021-01-23 10:59:12 +01:00
|
|
|
# match window background color
|
|
|
|
bgcolor = QPalette().window().color().name()
|
|
|
|
self.setStyleSheet("QTreeView { background: '%s'; }" % bgcolor)
|
|
|
|
|
2021-01-28 10:10:06 +01:00
|
|
|
def model(self) -> Union[FilterModel, SidebarModel]:
|
|
|
|
return super().model()
|
|
|
|
|
2021-01-23 10:59:12 +01:00
|
|
|
def refresh(self) -> None:
|
|
|
|
"Refresh list. No-op if sidebar is not visible."
|
|
|
|
if not self.isVisible():
|
|
|
|
return
|
|
|
|
|
2021-01-26 00:02:08 +01:00
|
|
|
def on_done(fut: Future):
|
|
|
|
root = fut.result()
|
|
|
|
model = SidebarModel(root)
|
2021-01-28 08:47:58 +01:00
|
|
|
|
2021-01-28 09:51:18 +01:00
|
|
|
# from PyQt5.QtTest import QAbstractItemModelTester
|
|
|
|
# tester = QAbstractItemModelTester(model)
|
2021-01-28 08:47:58 +01:00
|
|
|
|
2021-01-28 09:51:18 +01:00
|
|
|
self.setModel(model)
|
2021-01-29 02:20:15 +01:00
|
|
|
if self.current_search:
|
|
|
|
self.search_for(self.current_search)
|
|
|
|
else:
|
|
|
|
expand_where_necessary(model, self)
|
2021-01-26 00:02:08 +01:00
|
|
|
|
|
|
|
self.mw.taskman.run_in_background(self._root_tree, on_done)
|
2021-01-23 10:59:12 +01:00
|
|
|
|
2021-01-28 09:51:18 +01:00
|
|
|
def search_for(self, text: str):
|
|
|
|
if not text.strip():
|
2021-01-29 02:20:15 +01:00
|
|
|
self.current_search = None
|
2021-01-28 09:51:18 +01:00
|
|
|
self.refresh()
|
|
|
|
return
|
2021-01-28 10:10:06 +01:00
|
|
|
if not isinstance(self.model(), FilterModel):
|
|
|
|
filter_model = FilterModel(self)
|
2021-01-28 09:51:18 +01:00
|
|
|
filter_model.setSourceModel(self.model())
|
|
|
|
filter_model.setFilterCaseSensitivity(False) # type: ignore
|
|
|
|
filter_model.setRecursiveFilteringEnabled(True)
|
|
|
|
self.setModel(filter_model)
|
|
|
|
else:
|
|
|
|
filter_model = self.model()
|
|
|
|
|
2021-01-28 09:58:51 +01:00
|
|
|
self.current_search = text
|
2021-01-28 09:51:18 +01:00
|
|
|
# Without collapsing first, can be very slow. Surely there's
|
|
|
|
# a better way than this?
|
|
|
|
self.collapseAll()
|
|
|
|
filter_model.setFilterFixedString(text)
|
|
|
|
self.expandAll()
|
|
|
|
|
2021-01-28 10:10:06 +01:00
|
|
|
def drawRow(
|
|
|
|
self, painter: QPainter, options: QStyleOptionViewItem, idx: QModelIndex
|
|
|
|
):
|
|
|
|
if self.current_search is None:
|
|
|
|
return super().drawRow(painter, options, idx)
|
|
|
|
if not (item := self.model().item_for_index(idx)):
|
2021-01-28 09:58:51 +01:00
|
|
|
return super().drawRow(painter, options, idx)
|
2021-01-29 04:15:38 +01:00
|
|
|
if self.current_search.lower() in item.name.lower():
|
|
|
|
brush = QBrush(theme_manager.qcolor("suspended-bg"))
|
2021-01-28 09:58:51 +01:00
|
|
|
painter.save()
|
|
|
|
painter.fillRect(options.rect, brush)
|
|
|
|
painter.restore()
|
|
|
|
return super().drawRow(painter, options, idx)
|
|
|
|
|
2020-10-10 03:42:49 +02:00
|
|
|
def onClickCurrent(self) -> None:
|
|
|
|
idx = self.currentIndex()
|
2021-01-28 10:10:06 +01:00
|
|
|
if item := self.model().item_for_index(idx):
|
2020-10-10 03:42:49 +02:00
|
|
|
if item.onClick:
|
|
|
|
item.onClick()
|
|
|
|
|
|
|
|
def mouseReleaseEvent(self, event: QMouseEvent) -> None:
|
|
|
|
super().mouseReleaseEvent(event)
|
2021-01-07 01:07:51 +01:00
|
|
|
if event.button() == Qt.LeftButton:
|
|
|
|
self.onClickCurrent()
|
2020-10-10 03:42:49 +02:00
|
|
|
|
|
|
|
def keyPressEvent(self, event: QKeyEvent) -> None:
|
|
|
|
if event.key() in (Qt.Key_Return, Qt.Key_Enter):
|
|
|
|
self.onClickCurrent()
|
|
|
|
else:
|
|
|
|
super().keyPressEvent(event)
|
|
|
|
|
|
|
|
def onExpansion(self, idx: QModelIndex) -> None:
|
2021-01-28 09:58:51 +01:00
|
|
|
if self.current_search:
|
2021-01-28 09:51:18 +01:00
|
|
|
return
|
2020-10-10 03:42:49 +02:00
|
|
|
self._onExpansionChange(idx, True)
|
|
|
|
|
|
|
|
def onCollapse(self, idx: QModelIndex) -> None:
|
2021-01-28 09:58:51 +01:00
|
|
|
if self.current_search:
|
2021-01-28 09:51:18 +01:00
|
|
|
return
|
2020-10-10 03:42:49 +02:00
|
|
|
self._onExpansionChange(idx, False)
|
|
|
|
|
|
|
|
def _onExpansionChange(self, idx: QModelIndex, expanded: bool) -> None:
|
2021-01-28 10:10:06 +01:00
|
|
|
item = self.model().item_for_index(idx)
|
|
|
|
if item and item.expanded != expanded:
|
2020-10-10 03:42:49 +02:00
|
|
|
item.expanded = expanded
|
|
|
|
if item.onExpanded:
|
|
|
|
item.onExpanded(expanded)
|
|
|
|
|
2021-01-23 10:59:12 +01:00
|
|
|
# Tree building
|
|
|
|
###########################
|
2020-10-10 03:42:49 +02:00
|
|
|
|
2021-01-23 10:59:12 +01:00
|
|
|
def _root_tree(self) -> SidebarItem:
|
|
|
|
root = SidebarItem("", "", item_type=SidebarItemType.ROOT)
|
2020-10-10 03:42:49 +02:00
|
|
|
|
2021-01-23 10:59:12 +01:00
|
|
|
handled = gui_hooks.browser_will_build_tree(
|
|
|
|
False, root, SidebarStage.ROOT, self
|
|
|
|
)
|
|
|
|
if handled:
|
|
|
|
return root
|
|
|
|
|
|
|
|
for stage, builder in zip(
|
|
|
|
list(SidebarStage)[1:],
|
|
|
|
(
|
|
|
|
self._commonly_used_tree,
|
2021-01-29 13:10:31 +01:00
|
|
|
self._saved_searches_tree,
|
2021-01-23 10:59:12 +01:00
|
|
|
self._deck_tree,
|
|
|
|
self._notetype_tree,
|
|
|
|
self._tag_tree,
|
2021-01-20 01:26:53 +01:00
|
|
|
),
|
2021-01-23 10:59:12 +01:00
|
|
|
):
|
|
|
|
handled = gui_hooks.browser_will_build_tree(False, root, stage, self)
|
|
|
|
if not handled and builder:
|
|
|
|
builder(root)
|
|
|
|
|
|
|
|
return root
|
|
|
|
|
2021-01-29 13:10:31 +01:00
|
|
|
def _section_root(
|
|
|
|
self,
|
|
|
|
*,
|
|
|
|
root: SidebarItem,
|
|
|
|
name: TRValue,
|
|
|
|
icon: str,
|
|
|
|
collapse_key: ConfigBoolKeyValue,
|
|
|
|
type: Optional[SidebarItemType] = None,
|
|
|
|
) -> SidebarItem:
|
|
|
|
def update(expanded: bool):
|
|
|
|
self.col.set_config_bool(collapse_key, not expanded)
|
|
|
|
|
|
|
|
top = SidebarItem(
|
2021-01-29 13:50:29 +01:00
|
|
|
tr(name),
|
2021-01-29 13:10:31 +01:00
|
|
|
icon,
|
|
|
|
onExpanded=update,
|
|
|
|
expanded=not self.col.get_config_bool(collapse_key),
|
|
|
|
item_type=type,
|
|
|
|
)
|
|
|
|
root.addChild(top)
|
|
|
|
|
|
|
|
return top
|
|
|
|
|
2021-01-23 10:59:12 +01:00
|
|
|
def _commonly_used_tree(self, root: SidebarItem) -> None:
|
|
|
|
item = SidebarItem(
|
|
|
|
tr(TR.BROWSING_WHOLE_COLLECTION),
|
|
|
|
":/icons/collection.svg",
|
|
|
|
self._named_filter(NamedFilter.WHOLE_COLLECTION),
|
|
|
|
item_type=SidebarItemType.COLLECTION,
|
|
|
|
)
|
|
|
|
root.addChild(item)
|
|
|
|
item = SidebarItem(
|
|
|
|
tr(TR.BROWSING_CURRENT_DECK),
|
|
|
|
":/icons/deck.svg",
|
|
|
|
self._named_filter(NamedFilter.CURRENT_DECK),
|
|
|
|
item_type=SidebarItemType.CURRENT_DECK,
|
|
|
|
)
|
|
|
|
root.addChild(item)
|
|
|
|
|
2021-01-29 13:10:31 +01:00
|
|
|
def _saved_searches_tree(self, root: SidebarItem) -> None:
|
|
|
|
icon = ":/icons/heart.svg"
|
2021-01-23 10:59:12 +01:00
|
|
|
saved = self.col.get_config("savedFilters", {})
|
2021-01-29 13:10:31 +01:00
|
|
|
|
|
|
|
root = self._section_root(
|
|
|
|
root=root,
|
|
|
|
name=TR.BROWSING_SIDEBAR_SAVED_SEARCHES,
|
|
|
|
icon=icon,
|
|
|
|
collapse_key=ConfigBoolKey.COLLAPSE_FAVORITES,
|
|
|
|
type=SidebarItemType.SAVED_SEARCH_ROOT,
|
|
|
|
)
|
|
|
|
|
|
|
|
def on_click():
|
2021-01-29 13:50:29 +01:00
|
|
|
self.show_context_menu(root, None)
|
2021-01-29 13:10:31 +01:00
|
|
|
|
|
|
|
root.onClick = on_click
|
|
|
|
|
2021-01-23 10:59:12 +01:00
|
|
|
for name, filt in sorted(saved.items()):
|
|
|
|
item = SidebarItem(
|
|
|
|
name,
|
2021-01-29 13:10:31 +01:00
|
|
|
icon,
|
2021-01-23 10:59:12 +01:00
|
|
|
self._saved_filter(filt),
|
2021-01-29 13:10:31 +01:00
|
|
|
item_type=SidebarItemType.SAVED_SEARCH,
|
2021-01-23 10:59:12 +01:00
|
|
|
)
|
|
|
|
root.addChild(item)
|
|
|
|
|
|
|
|
def _tag_tree(self, root: SidebarItem) -> None:
|
2021-01-29 13:10:31 +01:00
|
|
|
icon = ":/icons/tag.svg"
|
2021-01-23 10:59:12 +01:00
|
|
|
|
|
|
|
def render(root: SidebarItem, nodes: Iterable[TagTreeNode], head="") -> None:
|
|
|
|
for node in nodes:
|
|
|
|
|
|
|
|
def toggle_expand():
|
|
|
|
full_name = head + node.name # pylint: disable=cell-var-from-loop
|
|
|
|
return lambda expanded: self.mw.col.tags.set_collapsed(
|
|
|
|
full_name, not expanded
|
|
|
|
)
|
|
|
|
|
|
|
|
item = SidebarItem(
|
|
|
|
node.name,
|
2021-01-29 13:10:31 +01:00
|
|
|
icon,
|
2021-01-23 10:59:12 +01:00
|
|
|
self._tag_filter(head + node.name),
|
|
|
|
toggle_expand(),
|
|
|
|
not node.collapsed,
|
|
|
|
item_type=SidebarItemType.TAG,
|
|
|
|
full_name=head + node.name,
|
|
|
|
)
|
|
|
|
root.addChild(item)
|
|
|
|
newhead = head + node.name + "::"
|
|
|
|
render(item, node.children, newhead)
|
|
|
|
|
2021-01-29 13:10:31 +01:00
|
|
|
tree = self.col.backend.tag_tree()
|
|
|
|
root = self._section_root(
|
|
|
|
root=root,
|
|
|
|
name=TR.BROWSING_SIDEBAR_TAGS,
|
|
|
|
icon=icon,
|
|
|
|
collapse_key=ConfigBoolKey.COLLAPSE_TAGS,
|
|
|
|
)
|
2021-01-23 10:59:12 +01:00
|
|
|
render(root, tree.children)
|
|
|
|
|
|
|
|
def _deck_tree(self, root: SidebarItem) -> None:
|
2021-01-29 13:10:31 +01:00
|
|
|
icon = ":/icons/deck.svg"
|
2021-01-23 10:59:12 +01:00
|
|
|
|
|
|
|
def render(root, nodes: Iterable[DeckTreeNode], head="") -> None:
|
|
|
|
for node in nodes:
|
|
|
|
|
|
|
|
def toggle_expand():
|
|
|
|
did = node.deck_id # pylint: disable=cell-var-from-loop
|
|
|
|
return lambda _: self.mw.col.decks.collapseBrowser(did)
|
|
|
|
|
|
|
|
item = SidebarItem(
|
|
|
|
node.name,
|
2021-01-29 13:10:31 +01:00
|
|
|
icon,
|
2021-01-23 10:59:12 +01:00
|
|
|
self._deck_filter(head + node.name),
|
|
|
|
toggle_expand(),
|
|
|
|
not node.collapsed,
|
|
|
|
item_type=SidebarItemType.DECK,
|
|
|
|
id=node.deck_id,
|
2021-01-26 23:35:40 +01:00
|
|
|
full_name=head + node.name,
|
2021-01-23 10:59:12 +01:00
|
|
|
)
|
|
|
|
root.addChild(item)
|
|
|
|
newhead = head + node.name + "::"
|
|
|
|
render(item, node.children, newhead)
|
|
|
|
|
2021-01-29 13:10:31 +01:00
|
|
|
tree = self.col.decks.deck_tree()
|
|
|
|
root = self._section_root(
|
|
|
|
root=root,
|
|
|
|
name=TR.BROWSING_SIDEBAR_DECKS,
|
|
|
|
icon=icon,
|
|
|
|
collapse_key=ConfigBoolKey.COLLAPSE_DECKS,
|
|
|
|
)
|
2021-01-23 10:59:12 +01:00
|
|
|
render(root, tree.children)
|
|
|
|
|
|
|
|
def _notetype_tree(self, root: SidebarItem) -> None:
|
2021-01-29 13:10:31 +01:00
|
|
|
icon = ":/icons/notetype.svg"
|
|
|
|
root = self._section_root(
|
|
|
|
root=root,
|
|
|
|
name=TR.BROWSING_SIDEBAR_NOTETYPES,
|
|
|
|
icon=icon,
|
|
|
|
collapse_key=ConfigBoolKey.COLLAPSE_NOTETYPES,
|
|
|
|
)
|
2021-01-23 10:59:12 +01:00
|
|
|
|
|
|
|
for nt in sorted(self.col.models.all(), key=lambda nt: nt["name"].lower()):
|
|
|
|
item = SidebarItem(
|
|
|
|
nt["name"],
|
2021-01-29 13:10:31 +01:00
|
|
|
icon=icon,
|
|
|
|
onClick=self._note_filter(nt["name"]),
|
2021-01-23 10:59:12 +01:00
|
|
|
item_type=SidebarItemType.NOTETYPE,
|
|
|
|
id=nt["id"],
|
|
|
|
)
|
|
|
|
|
|
|
|
for c, tmpl in enumerate(nt["tmpls"]):
|
|
|
|
child = SidebarItem(
|
|
|
|
tmpl["name"],
|
2021-01-29 13:10:31 +01:00
|
|
|
icon,
|
|
|
|
onClick=self._template_filter(nt["name"], c),
|
2021-01-23 10:59:12 +01:00
|
|
|
item_type=SidebarItemType.TEMPLATE,
|
2021-01-26 23:35:40 +01:00
|
|
|
full_name=nt["name"] + "::" + tmpl["name"],
|
2021-01-23 10:59:12 +01:00
|
|
|
)
|
|
|
|
item.addChild(child)
|
|
|
|
|
|
|
|
root.addChild(item)
|
|
|
|
|
|
|
|
def _named_filter(self, name: "FilterToSearchIn.NamedFilterValue") -> Callable:
|
|
|
|
return lambda: self.browser.update_search(
|
|
|
|
self.col.backend.filter_to_search(FilterToSearchIn(name=name))
|
|
|
|
)
|
|
|
|
|
|
|
|
def _tag_filter(self, tag: str) -> Callable:
|
|
|
|
return lambda: self.browser.update_search(
|
|
|
|
self.col.backend.filter_to_search(FilterToSearchIn(tag=tag))
|
|
|
|
)
|
|
|
|
|
|
|
|
def _deck_filter(self, deck: str) -> Callable:
|
|
|
|
return lambda: self.browser.update_search(
|
|
|
|
self.col.backend.filter_to_search(FilterToSearchIn(deck=deck))
|
|
|
|
)
|
|
|
|
|
|
|
|
def _note_filter(self, note: str) -> Callable:
|
|
|
|
return lambda: self.browser.update_search(
|
|
|
|
self.col.backend.filter_to_search(FilterToSearchIn(note=note))
|
|
|
|
)
|
|
|
|
|
|
|
|
def _template_filter(self, note: str, template: int) -> Callable:
|
|
|
|
return lambda: self.browser.update_search(
|
|
|
|
self.col.backend.filter_to_search(FilterToSearchIn(note=note)),
|
|
|
|
self.col.backend.filter_to_search(FilterToSearchIn(template=template)),
|
|
|
|
)
|
|
|
|
|
|
|
|
def _saved_filter(self, saved: str) -> Callable:
|
|
|
|
return lambda: self.browser.update_search(saved)
|
|
|
|
|
|
|
|
# Context menu actions
|
|
|
|
###########################
|
2020-10-10 03:42:49 +02:00
|
|
|
|
|
|
|
def onContextMenu(self, point: QPoint) -> None:
|
|
|
|
idx: QModelIndex = self.indexAt(point)
|
2021-01-28 10:10:06 +01:00
|
|
|
item = self.model().item_for_index(idx)
|
2020-10-10 03:42:49 +02:00
|
|
|
if not item:
|
|
|
|
return
|
2021-01-29 13:50:29 +01:00
|
|
|
self.show_context_menu(item, idx)
|
2021-01-29 13:10:31 +01:00
|
|
|
|
2021-01-29 13:50:29 +01:00
|
|
|
def show_context_menu(self, item: SidebarItem, idx: Optional[QModelIndex]):
|
|
|
|
m = QMenu()
|
|
|
|
|
|
|
|
if item.item_type in self.context_menus:
|
|
|
|
for action in self.context_menus[item.item_type]:
|
|
|
|
act_name = action[0]
|
|
|
|
act_func = action[1]
|
|
|
|
a = m.addAction(act_name)
|
|
|
|
qconnect(a.triggered, lambda _, func=act_func: func(item))
|
|
|
|
|
|
|
|
if idx:
|
|
|
|
self.maybe_add_tree_actions(m, item, idx)
|
|
|
|
|
|
|
|
if not m.children():
|
2020-10-10 03:42:49 +02:00
|
|
|
return
|
|
|
|
|
|
|
|
m.exec_(QCursor.pos())
|
|
|
|
|
2021-01-29 13:50:29 +01:00
|
|
|
def maybe_add_tree_actions(
|
|
|
|
self, menu: QMenu, item: SidebarItem, parent: QModelIndex
|
|
|
|
) -> None:
|
|
|
|
if self.current_search:
|
|
|
|
return
|
|
|
|
if not any(bool(c.children) for c in item.children):
|
|
|
|
return
|
|
|
|
|
|
|
|
def set_children_collapsed(collapsed: bool) -> None:
|
|
|
|
m = self.model()
|
|
|
|
self.setExpanded(parent, True)
|
|
|
|
for row in range(m.rowCount(parent)):
|
|
|
|
idx = m.index(row, 0, parent)
|
|
|
|
self.setExpanded(idx, not collapsed)
|
|
|
|
|
|
|
|
menu.addSeparator()
|
|
|
|
menu.addAction(
|
|
|
|
tr(TR.BROWSING_SIDEBAR_EXPAND_CHILDREN),
|
|
|
|
lambda: set_children_collapsed(False),
|
|
|
|
)
|
|
|
|
menu.addAction(
|
|
|
|
tr(TR.BROWSING_SIDEBAR_COLLAPSE_CHILDREN),
|
|
|
|
lambda: set_children_collapsed(True),
|
|
|
|
)
|
|
|
|
|
2021-01-04 05:13:20 +01:00
|
|
|
def rename_deck(self, item: "aqt.browser.SidebarItem") -> None:
|
2020-10-10 03:42:49 +02:00
|
|
|
deck = self.mw.col.decks.get(item.id)
|
|
|
|
old_name = deck["name"]
|
2020-11-17 08:42:43 +01:00
|
|
|
new_name = getOnlyText(tr(TR.DECKS_NEW_DECK_NAME), default=old_name)
|
2020-10-10 03:42:49 +02:00
|
|
|
new_name = new_name.replace('"', "")
|
|
|
|
if not new_name or new_name == old_name:
|
|
|
|
return
|
2021-01-03 08:30:14 +01:00
|
|
|
self.mw.checkpoint(tr(TR.ACTIONS_RENAME_DECK))
|
2020-10-10 03:42:49 +02:00
|
|
|
try:
|
|
|
|
self.mw.col.decks.rename(deck, new_name)
|
|
|
|
except DeckRenameError as e:
|
|
|
|
return showWarning(e.description)
|
2021-01-29 02:20:15 +01:00
|
|
|
self.refresh()
|
2021-01-03 01:09:42 +01:00
|
|
|
self.mw.deckBrowser.refresh()
|
2021-01-03 08:36:54 +01:00
|
|
|
|
2021-01-06 15:04:03 +01:00
|
|
|
def remove_tag(self, item: "aqt.browser.SidebarItem") -> None:
|
|
|
|
self.browser.editor.saveNow(lambda: self._remove_tag(item))
|
|
|
|
|
|
|
|
def _remove_tag(self, item: "aqt.browser.SidebarItem") -> None:
|
2021-01-09 02:49:10 +01:00
|
|
|
old_name = item.full_name
|
2021-01-06 15:04:03 +01:00
|
|
|
|
|
|
|
def do_remove():
|
|
|
|
self.mw.col.backend.clear_tag(old_name)
|
|
|
|
self.col.tags.rename_tag(old_name, "")
|
|
|
|
|
|
|
|
def on_done(fut: Future):
|
|
|
|
self.mw.requireReset(reason=ResetReason.BrowserRemoveTags, context=self)
|
|
|
|
self.browser.model.endReset()
|
|
|
|
fut.result()
|
2021-01-29 02:20:15 +01:00
|
|
|
self.refresh()
|
2021-01-06 15:04:03 +01:00
|
|
|
|
|
|
|
self.mw.checkpoint(tr(TR.ACTIONS_REMOVE_TAG))
|
|
|
|
self.browser.model.beginReset()
|
|
|
|
self.mw.taskman.run_in_background(do_remove, on_done)
|
|
|
|
|
2021-01-04 05:13:20 +01:00
|
|
|
def rename_tag(self, item: "aqt.browser.SidebarItem") -> None:
|
2021-01-03 08:36:54 +01:00
|
|
|
self.browser.editor.saveNow(lambda: self._rename_tag(item))
|
|
|
|
|
2021-01-04 05:13:20 +01:00
|
|
|
def _rename_tag(self, item: "aqt.browser.SidebarItem") -> None:
|
2021-01-09 02:49:10 +01:00
|
|
|
old_name = item.full_name
|
2021-01-03 08:36:54 +01:00
|
|
|
new_name = getOnlyText(tr(TR.ACTIONS_NEW_NAME), default=old_name)
|
|
|
|
if new_name == old_name or not new_name:
|
|
|
|
return
|
2021-01-04 05:13:20 +01:00
|
|
|
|
|
|
|
def do_rename():
|
2021-01-06 15:04:03 +01:00
|
|
|
self.mw.col.backend.clear_tag(old_name)
|
2021-01-04 05:13:20 +01:00
|
|
|
return self.col.tags.rename_tag(old_name, new_name)
|
|
|
|
|
|
|
|
def on_done(fut: Future):
|
|
|
|
self.mw.requireReset(reason=ResetReason.BrowserAddTags, context=self)
|
|
|
|
self.browser.model.endReset()
|
|
|
|
|
|
|
|
count = fut.result()
|
|
|
|
if not count:
|
|
|
|
showInfo(tr(TR.BROWSING_TAG_RENAME_WARNING_EMPTY))
|
|
|
|
return
|
|
|
|
|
2021-01-29 02:20:15 +01:00
|
|
|
self.refresh()
|
2021-01-04 05:13:20 +01:00
|
|
|
|
2021-01-03 08:36:54 +01:00
|
|
|
self.mw.checkpoint(tr(TR.ACTIONS_RENAME_TAG))
|
|
|
|
self.browser.model.beginReset()
|
2021-01-04 05:13:20 +01:00
|
|
|
self.mw.taskman.run_in_background(do_rename, on_done)
|
2021-01-04 11:30:41 +01:00
|
|
|
|
|
|
|
def delete_deck(self, item: "aqt.browser.SidebarItem") -> None:
|
|
|
|
self.browser.editor.saveNow(lambda: self._delete_deck(item))
|
|
|
|
|
|
|
|
def _delete_deck(self, item: "aqt.browser.SidebarItem") -> None:
|
|
|
|
did = item.id
|
|
|
|
if self.mw.deckBrowser.ask_delete_deck(did):
|
|
|
|
|
|
|
|
def do_delete():
|
|
|
|
return self.mw.col.decks.rem(did, True)
|
|
|
|
|
|
|
|
def on_done(fut: Future):
|
2021-01-04 11:36:45 +01:00
|
|
|
self.mw.requireReset(reason=ResetReason.BrowserDeleteDeck, context=self)
|
2021-01-04 11:30:41 +01:00
|
|
|
self.browser.search()
|
|
|
|
self.browser.model.endReset()
|
2021-01-29 02:20:15 +01:00
|
|
|
self.refresh()
|
2021-01-04 11:30:41 +01:00
|
|
|
res = fut.result() # Required to check for errors
|
|
|
|
|
|
|
|
self.mw.checkpoint(tr(TR.DECKS_DELETE_DECK))
|
|
|
|
self.browser.model.beginReset()
|
|
|
|
self.mw.taskman.run_in_background(do_delete, on_done)
|
2021-01-20 01:26:53 +01:00
|
|
|
|
2021-01-29 14:05:30 +01:00
|
|
|
def remove_saved_search(self, item: "aqt.browser.SidebarItem") -> None:
|
|
|
|
name = item.name
|
|
|
|
if not askUser(tr(TR.BROWSING_REMOVE_FROM_YOUR_SAVED_SEARCHES, val=name)):
|
|
|
|
return
|
|
|
|
conf = self.col.get_config("savedFilters")
|
|
|
|
del conf[name]
|
|
|
|
self.col.set_config("savedFilters", conf)
|
|
|
|
self.refresh()
|
2021-01-20 01:26:53 +01:00
|
|
|
|
2021-01-29 14:05:30 +01:00
|
|
|
def rename_saved_search(self, item: "aqt.browser.SidebarItem") -> None:
|
|
|
|
old = item.name
|
|
|
|
conf = self.col.get_config("savedFilters")
|
|
|
|
try:
|
|
|
|
filt = conf[old]
|
|
|
|
except KeyError:
|
|
|
|
return
|
|
|
|
new = getOnlyText(tr(TR.ACTIONS_NEW_NAME), default=old)
|
|
|
|
if new == old or not new:
|
|
|
|
return
|
|
|
|
conf[new] = filt
|
|
|
|
del conf[old]
|
|
|
|
self.col.set_config("savedFilters", conf)
|
|
|
|
self.refresh()
|
|
|
|
|
|
|
|
def save_current_search(self, _item=None) -> None:
|
|
|
|
try:
|
|
|
|
filt = self.col.backend.normalize_search(
|
|
|
|
self.browser.form.searchEdit.lineEdit().text()
|
|
|
|
)
|
|
|
|
except InvalidInput as e:
|
|
|
|
show_invalid_search_error(e)
|
|
|
|
else:
|
|
|
|
name = getOnlyText(tr(TR.BROWSING_PLEASE_GIVE_YOUR_FILTER_A_NAME))
|
|
|
|
if not name:
|
|
|
|
return
|
|
|
|
conf = self.col.get_config("savedFilters")
|
|
|
|
conf[name] = filt
|
|
|
|
self.col.set_config("savedFilters", conf)
|
|
|
|
self.refresh()
|
2021-01-20 00:00:53 +01:00
|
|
|
|
|
|
|
def manage_notetype(self, item: "aqt.browser.SidebarItem") -> None:
|
2021-01-22 03:56:39 +01:00
|
|
|
Models(
|
|
|
|
self.mw, parent=self.browser, fromMain=True, selected_notetype_id=item.id
|
|
|
|
)
|