mirror of
https://github.com/kovidgoyal/calibre.git
synced 2025-07-09 03:04:10 -04:00
Add Sort By context menu entry
A new "Sort By" action for the right click menu. This allows sorting on all columns in the library, not just the visible columns. To use it go to Preferences->Toolbars and add it to "The context menu for books in the calibre library"
This commit is contained in:
parent
c4a82de4c7
commit
8b5e9ed035
@ -912,6 +912,10 @@ class ActionPickRandom(InterfaceActionBase):
|
|||||||
actual_plugin = 'calibre.gui2.actions.random:PickRandomAction'
|
actual_plugin = 'calibre.gui2.actions.random:PickRandomAction'
|
||||||
description = _('Choose a random book from your calibre library')
|
description = _('Choose a random book from your calibre library')
|
||||||
|
|
||||||
|
class ActionSortBy(InterfaceActionBase):
|
||||||
|
name = 'Sort By'
|
||||||
|
actual_plugin = 'calibre.gui2.actions.sort:SortByAction'
|
||||||
|
description = _('Sort the list of books')
|
||||||
|
|
||||||
class ActionStore(InterfaceActionBase):
|
class ActionStore(InterfaceActionBase):
|
||||||
name = 'Store'
|
name = 'Store'
|
||||||
@ -943,7 +947,7 @@ plugins += [ActionAdd, ActionFetchAnnotations, ActionGenerateCatalog,
|
|||||||
ActionSendToDevice, ActionHelp, ActionPreferences, ActionSimilarBooks,
|
ActionSendToDevice, ActionHelp, ActionPreferences, ActionSimilarBooks,
|
||||||
ActionAddToLibrary, ActionEditCollections, ActionMatchBooks, ActionChooseLibrary,
|
ActionAddToLibrary, ActionEditCollections, ActionMatchBooks, ActionChooseLibrary,
|
||||||
ActionCopyToLibrary, ActionTweakEpub, ActionNextMatch, ActionStore,
|
ActionCopyToLibrary, ActionTweakEpub, ActionNextMatch, ActionStore,
|
||||||
ActionPluginUpdater, ActionPickRandom, ActionEditToC]
|
ActionPluginUpdater, ActionPickRandom, ActionEditToC, ActionSortBy]
|
||||||
|
|
||||||
# }}}
|
# }}}
|
||||||
|
|
||||||
|
79
src/calibre/gui2/actions/sort.py
Normal file
79
src/calibre/gui2/actions/sort.py
Normal file
@ -0,0 +1,79 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
# vim:fileencoding=utf-8
|
||||||
|
from __future__ import (unicode_literals, division, absolute_import,
|
||||||
|
print_function)
|
||||||
|
|
||||||
|
__license__ = 'GPL v3'
|
||||||
|
__copyright__ = '2013, Kovid Goyal <kovid at kovidgoyal.net>'
|
||||||
|
|
||||||
|
from PyQt4.Qt import QToolButton, QAction, pyqtSignal, QIcon
|
||||||
|
|
||||||
|
from calibre.gui2.actions import InterfaceAction
|
||||||
|
from calibre.utils.icu import sort_key
|
||||||
|
|
||||||
|
class SortAction(QAction):
|
||||||
|
|
||||||
|
sort_requested = pyqtSignal(object, object)
|
||||||
|
|
||||||
|
def __init__(self, text, key, ascending, parent):
|
||||||
|
QAction.__init__(self, text, parent)
|
||||||
|
self.key, self.ascending = key, ascending
|
||||||
|
self.triggered.connect(self)
|
||||||
|
|
||||||
|
def __call__(self):
|
||||||
|
self.sort_requested(self.key, self.ascending)
|
||||||
|
|
||||||
|
class SortByAction(InterfaceAction):
|
||||||
|
|
||||||
|
name = 'Sort By'
|
||||||
|
action_spec = (_('Sort By'), 'arrow-up.png', _('Sort the list of books'), None)
|
||||||
|
action_type = 'current'
|
||||||
|
popup_type = QToolButton.InstantPopup
|
||||||
|
action_add_menu = True
|
||||||
|
dont_add_to = frozenset([
|
||||||
|
'toolbar', 'toolbar-device', 'context-menu-device', 'toolbar-child',
|
||||||
|
'menubar', 'menubar-device', 'context-menu-cover-browser'])
|
||||||
|
|
||||||
|
def genesis(self):
|
||||||
|
self.sorted_icon = QIcon(I('ok.png'))
|
||||||
|
|
||||||
|
def location_selected(self, loc):
|
||||||
|
self.qaction.setEnabled(loc == 'library')
|
||||||
|
|
||||||
|
def update_menu(self):
|
||||||
|
menu = self.qaction.menu()
|
||||||
|
for action in menu.actions():
|
||||||
|
action.sort_requested.disconnect()
|
||||||
|
menu.clear()
|
||||||
|
lv = self.gui.library_view
|
||||||
|
m = lv.model()
|
||||||
|
db = m.db
|
||||||
|
try:
|
||||||
|
sort_col, order = m.sorted_on
|
||||||
|
except TypeError:
|
||||||
|
sort_col, order = 'date', True
|
||||||
|
fm = db.field_metadata
|
||||||
|
name_map = {fm[k]['name']:k for k in fm.sortable_field_keys() if fm[k]['name']}
|
||||||
|
self._sactions = []
|
||||||
|
for name in sorted(name_map, key=sort_key):
|
||||||
|
key = name_map[name]
|
||||||
|
if key == 'title':
|
||||||
|
continue
|
||||||
|
if key == 'sort':
|
||||||
|
name = _('Title')
|
||||||
|
if key == 'ondevice' and self.gui.device_connected is None:
|
||||||
|
continue
|
||||||
|
ascending = True
|
||||||
|
if key == sort_col:
|
||||||
|
name = _('%s [reverse current sort]') % name
|
||||||
|
ascending = not order
|
||||||
|
sac = SortAction(name, key, ascending, menu)
|
||||||
|
if key == sort_col:
|
||||||
|
sac.setIcon(self.sorted_icon)
|
||||||
|
sac.sort_requested.connect(self.sort_requested)
|
||||||
|
menu.addAction(sac)
|
||||||
|
|
||||||
|
def sort_requested(self, key, ascending):
|
||||||
|
self.gui.library_view.sort_by_named_field(key, ascending)
|
||||||
|
|
||||||
|
|
@ -15,9 +15,10 @@ from Queue import Queue
|
|||||||
from functools import wraps, partial
|
from functools import wraps, partial
|
||||||
|
|
||||||
from PyQt4.Qt import (
|
from PyQt4.Qt import (
|
||||||
QListView, QSize, QStyledItemDelegate, QModelIndex, Qt, QImage, pyqtSignal, QTimer,
|
QListView, QSize, QStyledItemDelegate, QModelIndex, Qt, QImage, pyqtSignal,
|
||||||
QPalette, QColor, QItemSelection, QPixmap, QMenu, QApplication, QMimeData, QIcon,
|
QTimer, QPalette, QColor, QItemSelection, QPixmap, QMenu, QApplication,
|
||||||
QUrl, QDrag, QPoint, QPainter, QRect, pyqtProperty, QPropertyAnimation, QEasingCurve)
|
QMimeData, QUrl, QDrag, QPoint, QPainter, QRect, pyqtProperty,
|
||||||
|
QPropertyAnimation, QEasingCurve)
|
||||||
|
|
||||||
from calibre import fit_image
|
from calibre import fit_image
|
||||||
from calibre.gui2 import gprefs, config
|
from calibre.gui2 import gprefs, config
|
||||||
@ -215,7 +216,6 @@ class AlternateViews(object):
|
|||||||
view.setModel(self.main_view._model)
|
view.setModel(self.main_view._model)
|
||||||
view.selectionModel().currentChanged.connect(self.slave_current_changed)
|
view.selectionModel().currentChanged.connect(self.slave_current_changed)
|
||||||
view.selectionModel().selectionChanged.connect(self.slave_selection_changed)
|
view.selectionModel().selectionChanged.connect(self.slave_selection_changed)
|
||||||
view.sort_requested.connect(self.main_view.sort_by_named_field)
|
|
||||||
view.files_dropped.connect(self.main_view.files_dropped)
|
view.files_dropped.connect(self.main_view.files_dropped)
|
||||||
|
|
||||||
def show_view(self, key=None):
|
def show_view(self, key=None):
|
||||||
@ -440,7 +440,6 @@ def join_with_timeout(q, timeout=2):
|
|||||||
class GridView(QListView):
|
class GridView(QListView):
|
||||||
|
|
||||||
update_item = pyqtSignal(object)
|
update_item = pyqtSignal(object)
|
||||||
sort_requested = pyqtSignal(object, object)
|
|
||||||
files_dropped = pyqtSignal(object)
|
files_dropped = pyqtSignal(object)
|
||||||
|
|
||||||
def __init__(self, parent):
|
def __init__(self, parent):
|
||||||
@ -648,31 +647,17 @@ class GridView(QListView):
|
|||||||
|
|
||||||
def contextMenuEvent(self, event):
|
def contextMenuEvent(self, event):
|
||||||
if self.context_menu is not None:
|
if self.context_menu is not None:
|
||||||
lv = self.gui.library_view
|
|
||||||
menu = self._temp_menu = QMenu(self)
|
menu = self._temp_menu = QMenu(self)
|
||||||
sm = QMenu(_('Sort by'), menu)
|
sac = self.gui.iactions['Sort By']
|
||||||
db = self.model().db
|
sort_added = tuple(ac for ac in self.context_menu.actions() if ac is sac.qaction)
|
||||||
for col in lv.visible_columns:
|
if not sort_added:
|
||||||
m = db.metadata_for_field(col)
|
menu.addAction(sac.qaction)
|
||||||
last = self.model().sorted_on
|
|
||||||
ascending = True
|
|
||||||
extra = ''
|
|
||||||
if last[0] == col:
|
|
||||||
ascending = not last[1]
|
|
||||||
extra = ' [%s]' % _('reverse current sort')
|
|
||||||
ac = sm.addAction('%s%s' % (m.get('name', col), extra), partial(self.do_sort, col, ascending))
|
|
||||||
if last[0] == col:
|
|
||||||
ac.setIcon(QIcon(I('ok.png')))
|
|
||||||
|
|
||||||
for ac in self.context_menu.actions():
|
for ac in self.context_menu.actions():
|
||||||
menu.addAction(ac)
|
menu.addAction(ac)
|
||||||
menu.addMenu(sm).setIcon(QIcon(I('arrow-up.png')))
|
sac.update_menu()
|
||||||
menu.popup(event.globalPos())
|
menu.popup(event.globalPos())
|
||||||
event.accept()
|
event.accept()
|
||||||
|
|
||||||
def do_sort(self, column, ascending):
|
|
||||||
self.sort_requested.emit(column, ascending)
|
|
||||||
|
|
||||||
def get_selected_ids(self):
|
def get_selected_ids(self):
|
||||||
m = self.model()
|
m = self.model()
|
||||||
return [m.id(i) for i in self.selectionModel().selectedIndexes()]
|
return [m.id(i) for i in self.selectionModel().selectedIndexes()]
|
||||||
|
@ -157,6 +157,7 @@ class BooksView(QTableView): # {{{
|
|||||||
|
|
||||||
def __init__(self, parent, modelcls=BooksModel, use_edit_metadata_dialog=True):
|
def __init__(self, parent, modelcls=BooksModel, use_edit_metadata_dialog=True):
|
||||||
QTableView.__init__(self, parent)
|
QTableView.__init__(self, parent)
|
||||||
|
self.gui = parent
|
||||||
self.setProperty('highlight_current_item', 150)
|
self.setProperty('highlight_current_item', 150)
|
||||||
self.row_sizing_done = False
|
self.row_sizing_done = False
|
||||||
self.alternate_views = AlternateViews(self)
|
self.alternate_views = AlternateViews(self)
|
||||||
@ -713,6 +714,10 @@ class BooksView(QTableView): # {{{
|
|||||||
self.edit_collections_action = edit_collections_action
|
self.edit_collections_action = edit_collections_action
|
||||||
|
|
||||||
def contextMenuEvent(self, event):
|
def contextMenuEvent(self, event):
|
||||||
|
sac = self.gui.iactions['Sort By']
|
||||||
|
sort_added = tuple(ac for ac in self.context_menu.actions() if ac is sac.qaction)
|
||||||
|
if sort_added:
|
||||||
|
sac.update_menu()
|
||||||
self.context_menu.popup(event.globalPos())
|
self.context_menu.popup(event.globalPos())
|
||||||
event.accept()
|
event.accept()
|
||||||
# }}}
|
# }}}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user