Read indicators, progress tracking, database closing
This commit is contained in:
60
__main__.py
60
__main__.py
@@ -87,6 +87,9 @@ class MainUI(QtWidgets.QMainWindow, mainwindow.Ui_MainWindow):
|
||||
self.viewModel = None
|
||||
self.lib_ref = Library(self)
|
||||
|
||||
# Application wide temporary directory
|
||||
self.temp_dir = QtCore.QTemporaryDir()
|
||||
|
||||
# Library toolbar
|
||||
self.libraryToolBar = LibraryToolBar(self)
|
||||
self.libraryToolBar.addButton.triggered.connect(self.add_books)
|
||||
@@ -128,15 +131,14 @@ class MainUI(QtWidgets.QMainWindow, mainwindow.Ui_MainWindow):
|
||||
# TODO
|
||||
# It's possible to add a widget to the Library tab here
|
||||
self.tabWidget.tabBar().setTabButton(0, QtWidgets.QTabBar.RightSide, None)
|
||||
self.tabWidget.tabCloseRequested.connect(self.close_tab)
|
||||
self.tabWidget.tabCloseRequested.connect(self.tab_close)
|
||||
|
||||
# ListView
|
||||
# self.listView.setSpacing(0)
|
||||
self.listView.setGridSize(QtCore.QSize(175, 240))
|
||||
self.listView.setMouseTracking(True)
|
||||
self.listView.verticalScrollBar().setSingleStep(7)
|
||||
self.listView.doubleClicked.connect(self.list_doubleclick)
|
||||
self.listView.setItemDelegate(LibraryDelegate())
|
||||
self.listView.setItemDelegate(LibraryDelegate(self.temp_dir.path()))
|
||||
self.reload_listview()
|
||||
|
||||
# Keyboard shortcuts
|
||||
@@ -262,7 +264,7 @@ class MainUI(QtWidgets.QMainWindow, mainwindow.Ui_MainWindow):
|
||||
self.bookToolBar.tocBox.clear()
|
||||
self.bookToolBar.tocBox.addItems(current_toc)
|
||||
if current_position:
|
||||
self.bookToolBar.tocBox.setCurrentIndex(current_position)
|
||||
self.bookToolBar.tocBox.setCurrentIndex(current_position['current_chapter'] - 1)
|
||||
self.bookToolBar.tocBox.blockSignals(False)
|
||||
|
||||
self.format_contentView()
|
||||
@@ -270,18 +272,46 @@ class MainUI(QtWidgets.QMainWindow, mainwindow.Ui_MainWindow):
|
||||
self.statusMessage.setText(
|
||||
current_author + ' - ' + current_title)
|
||||
|
||||
def tab_close(self, tab_index):
|
||||
self.database_update_position(tab_index)
|
||||
temp_dir = self.tabWidget.widget(tab_index).metadata['temp_dir']
|
||||
if temp_dir:
|
||||
shutil.rmtree(temp_dir)
|
||||
self.tabWidget.removeTab(tab_index)
|
||||
|
||||
def set_toc_position(self, event=None):
|
||||
self.tabWidget.widget(
|
||||
self.tabWidget.currentIndex()).metadata[
|
||||
'position'] = event
|
||||
|
||||
chapter_name = self.bookToolBar.tocBox.currentText()
|
||||
|
||||
current_tab = self.tabWidget.widget(self.tabWidget.currentIndex())
|
||||
required_content = current_tab.metadata['content'][chapter_name]
|
||||
|
||||
# We're also updating the underlying model to have real-time
|
||||
# updates on the read status
|
||||
# Find index of the model item that corresponds to the tab
|
||||
start_index = self.viewModel.index(0, 0)
|
||||
matching_item = self.viewModel.match(
|
||||
start_index,
|
||||
QtCore.Qt.UserRole + 6,
|
||||
current_tab.metadata['hash'],
|
||||
1, QtCore.Qt.MatchExactly)
|
||||
if matching_item:
|
||||
model_row = matching_item[0].row()
|
||||
model_index = self.viewModel.index(model_row, 0)
|
||||
|
||||
current_tab.metadata[
|
||||
'position']['current_chapter'] = self.bookToolBar.tocBox.currentIndex() + 1
|
||||
self.viewModel.setData(
|
||||
model_index, current_tab.metadata['position'], QtCore.Qt.UserRole + 7)
|
||||
|
||||
current_tab.contentView.verticalScrollBar().setValue(0)
|
||||
current_tab.contentView.setHtml(required_content)
|
||||
|
||||
def database_update_position(self, tab_index):
|
||||
tab_metadata = self.tabWidget.widget(tab_index).metadata
|
||||
file_hash = tab_metadata['hash']
|
||||
position = tab_metadata['position']
|
||||
database.DatabaseFunctions(
|
||||
self.database_path).modify_position(file_hash, position)
|
||||
|
||||
def set_fullscreen(self):
|
||||
current_tab = self.tabWidget.currentIndex()
|
||||
current_tab_widget = self.tabWidget.widget(current_tab)
|
||||
@@ -294,9 +324,9 @@ class MainUI(QtWidgets.QMainWindow, mainwindow.Ui_MainWindow):
|
||||
|
||||
def list_doubleclick(self, myindex):
|
||||
index = self.listView.model().index(myindex.row(), 0)
|
||||
state = self.listView.model().data(index, QtCore.Qt.UserRole + 5)
|
||||
file_exists = self.listView.model().data(index, QtCore.Qt.UserRole + 5)
|
||||
|
||||
if state == 'deleted':
|
||||
if not file_exists:
|
||||
return
|
||||
|
||||
metadata = self.listView.model().data(index, QtCore.Qt.UserRole + 3)
|
||||
@@ -316,12 +346,6 @@ class MainUI(QtWidgets.QMainWindow, mainwindow.Ui_MainWindow):
|
||||
self.tabWidget.setCurrentWidget(tab_ref)
|
||||
self.format_contentView()
|
||||
|
||||
def close_tab(self, tab_index):
|
||||
temp_dir = self.tabWidget.widget(tab_index).metadata['temp_dir']
|
||||
if temp_dir:
|
||||
shutil.rmtree(temp_dir)
|
||||
self.tabWidget.removeTab(tab_index)
|
||||
|
||||
def get_color(self):
|
||||
signal_sender = self.sender().objectName()
|
||||
profile_index = self.bookToolBar.profileBox.currentIndex()
|
||||
@@ -427,10 +451,12 @@ class MainUI(QtWidgets.QMainWindow, mainwindow.Ui_MainWindow):
|
||||
def closeEvent(self, event=None):
|
||||
# All tabs must be iterated upon here
|
||||
for i in range(1, self.tabWidget.count()):
|
||||
self.database_update_position(i)
|
||||
tab_metadata = self.tabWidget.widget(i).metadata
|
||||
if tab_metadata['temp_dir']:
|
||||
shutil.rmtree(tab_metadata['temp_dir'])
|
||||
|
||||
self.temp_dir.remove()
|
||||
Settings(self).save_settings()
|
||||
QtWidgets.qApp.exit()
|
||||
|
||||
|
21
database.py
21
database.py
@@ -1,7 +1,8 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
import sqlite3
|
||||
import os
|
||||
import pickle
|
||||
import sqlite3
|
||||
|
||||
|
||||
class DatabaseInit:
|
||||
@@ -17,7 +18,7 @@ class DatabaseInit:
|
||||
self.database.execute(
|
||||
"CREATE TABLE books \
|
||||
(id INTEGER PRIMARY KEY, Title TEXT, Author TEXT, Year INTEGER, \
|
||||
Path TEXT, Position TEXT, ISBN TEXT, Tags TEXT, Hash TEXT, CoverImage BLOB)")
|
||||
Path TEXT, Position BLOB, ISBN TEXT, Tags TEXT, Hash TEXT, CoverImage BLOB)")
|
||||
self.database.execute(
|
||||
"CREATE TABLE cache \
|
||||
(id INTEGER PRIMARY KEY, Name TEXT, Path TEXT, CachedDict BLOB)")
|
||||
@@ -62,6 +63,7 @@ class DatabaseFunctions:
|
||||
path, isbn, book_hash, sqlite3.Binary(cover)])
|
||||
|
||||
self.database.commit()
|
||||
self.close_database()
|
||||
|
||||
def fetch_data(self, columns, table, selection_criteria, equivalence, fetch_one=False):
|
||||
# columns is a tuple that will be passed as a comma separated list
|
||||
@@ -111,6 +113,16 @@ class DatabaseFunctions:
|
||||
except KeyError:
|
||||
print('SQLite is in rebellion, Commander')
|
||||
|
||||
self.close_database()
|
||||
|
||||
def modify_position(self, file_hash, position):
|
||||
pickled_position = pickle.dumps(position)
|
||||
|
||||
sql_command = "UPDATE books SET Position = ? WHERE Hash = ?"
|
||||
self.database.execute(sql_command, [sqlite3.Binary(pickled_position), file_hash])
|
||||
self.database.commit()
|
||||
self.close_database()
|
||||
|
||||
def delete_from_database(self, file_hashes):
|
||||
# file_hashes is expected as a list that will be iterated upon
|
||||
# This should enable multiple deletion
|
||||
@@ -119,3 +131,8 @@ class DatabaseFunctions:
|
||||
self.database.execute(
|
||||
f"DELETE FROM books WHERE Hash = '{i}'")
|
||||
self.database.commit()
|
||||
self.close_database()
|
||||
|
||||
def close_database(self):
|
||||
self.database.execute("VACUUM")
|
||||
self.database.close()
|
||||
|
27
pie_chart.py
27
pie_chart.py
@@ -1,13 +1,15 @@
|
||||
# Modified from: http://drumcoder.co.uk/blog/2010/nov/16/python-code-generate-svg-pie-chart/
|
||||
|
||||
import os
|
||||
import math
|
||||
|
||||
class GeneratePie():
|
||||
def __init__(self, progress_percent):
|
||||
def __init__(self, progress_percent, temp_dir=None):
|
||||
self.progress_percent = int(progress_percent)
|
||||
self.temp_dir = temp_dir
|
||||
|
||||
def generate(self):
|
||||
lSlices = (100 - self.progress_percent, self.progress_percent) # percentages to show in pie
|
||||
lSlices = (self.progress_percent, 100 - self.progress_percent) # percentages to show in pie
|
||||
|
||||
lOffsetX = 150
|
||||
lOffsetY = 150
|
||||
@@ -57,7 +59,7 @@ class GeneratePie():
|
||||
|
||||
lPath = "%s %s %s" % (lLineOne, lArc, lLineTwo)
|
||||
lGradient = GRADIENTS[lIndex]
|
||||
lSvgPath += "<path d='%s' style='stroke:#2c2c2c; fill:url(#%s);'/>" % (
|
||||
lSvgPath += "<path d='%s' style='stroke:#c579be; fill:url(#%s);'/>" % (
|
||||
lPath, lGradient)
|
||||
lIndex += 1
|
||||
|
||||
@@ -66,20 +68,27 @@ class GeneratePie():
|
||||
xmlns:xlink="http://www.w3.org/1999/xlink">
|
||||
<defs>
|
||||
<radialGradient id="myRadialGradientGreen" r="65%%" cx="0" cy="0" spreadMethod="pad">
|
||||
<stop offset="0%%" stop-color="#2c2c2c" stop-opacity="1"/>
|
||||
<stop offset="100%%" stop-color="#2c2c2c" stop-opacity="1" />
|
||||
<stop offset="0%%" stop-color="#c579be" stop-opacity="1"/>
|
||||
<stop offset="100%%" stop-color="#c579be" stop-opacity="1" />
|
||||
</radialGradient>
|
||||
</defs>
|
||||
<defs>
|
||||
<radialGradient id="myRadialGradientOrange" r="65%%" cx="0" cy="0" spreadMethod="pad">
|
||||
<stop offset="0%%" stop-color="#4caf50" stop-opacity="1"/>
|
||||
<stop offset="100%%" stop-color="#4caf50" stop-opacity="1" />
|
||||
<stop offset="0%%" stop-color="#6c4268" stop-opacity="1"/>
|
||||
<stop offset="100%%" stop-color="#6c4268" stop-opacity="1" />
|
||||
</radialGradient>
|
||||
</defs>
|
||||
|
||||
%s
|
||||
<!-- <circle cx="%d" cy="%d" r="100" style="stroke:#4caf50; fill:none;"/> -->
|
||||
<!-- <circle cx="%d" cy="%d" r="100" style="stroke:#6c4268; fill:none;"/> -->
|
||||
</svg>
|
||||
""" % (lSvgPath, lOffsetX, lOffsetY)
|
||||
|
||||
return lSvg
|
||||
|
||||
if self.temp_dir:
|
||||
svg_path = os.path.join(self.temp_dir, 'lector_progress.svg')
|
||||
lFile = open(svg_path, 'w')
|
||||
lFile.write(lSvg)
|
||||
lFile.close()
|
||||
else:
|
||||
return lSvg
|
||||
|
@@ -5,6 +5,7 @@
|
||||
# See if you want to include a hash of the book's name and author
|
||||
|
||||
import os
|
||||
import pickle
|
||||
import hashlib
|
||||
from multiprocessing.dummy import Pool
|
||||
|
||||
@@ -18,6 +19,7 @@ import database
|
||||
# get_cover_image()
|
||||
# get_isbn()
|
||||
# get_contents() - Should return a tuple with 0: TOC 1: Deletable temp_directory
|
||||
|
||||
from parsers.epub import ParseEPUB
|
||||
from parsers.cbz import ParseCBZ
|
||||
|
||||
@@ -57,7 +59,12 @@ class BookSorter:
|
||||
{'Hash': file_hash},
|
||||
'EQUALS',
|
||||
True)
|
||||
return position
|
||||
|
||||
if position:
|
||||
position_dict = pickle.loads(position)
|
||||
return position_dict
|
||||
else:
|
||||
return None
|
||||
|
||||
def read_book(self, filename):
|
||||
# filename is expected as a string containg the
|
||||
|
@@ -1,6 +1,7 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
import os
|
||||
import pickle
|
||||
import database
|
||||
from PyQt5 import QtWidgets, QtGui, QtCore
|
||||
|
||||
@@ -11,12 +12,11 @@ class Library:
|
||||
self.proxy_model = None
|
||||
|
||||
def generate_model(self):
|
||||
# TODO
|
||||
# Use QItemdelegates to show book read progress
|
||||
|
||||
# The QlistView widget needs to be populated
|
||||
# with a model that inherits from QStandardItemModel
|
||||
self.parent_window.viewModel = QtGui.QStandardItemModel()
|
||||
# self.parent_window.viewModel = QtGui.QStandardItemModel()
|
||||
self.parent_window.viewModel = MyAbsModel()
|
||||
|
||||
books = database.DatabaseFunctions(
|
||||
self.parent_window.database_path).fetch_data(
|
||||
('*',),
|
||||
@@ -37,21 +37,19 @@ class Library:
|
||||
path = i[4]
|
||||
tags = i[6]
|
||||
cover = i[9]
|
||||
progress = None # TODO
|
||||
# Leave at None for an untouched book
|
||||
# 'completed' for a completed book
|
||||
# whatever else is here can be used
|
||||
# to remember position
|
||||
# Maybe get from the position param
|
||||
|
||||
position = i[5]
|
||||
if position:
|
||||
position = pickle.loads(position)
|
||||
|
||||
all_metadata = {
|
||||
'title': i[1],
|
||||
'author': i[2],
|
||||
'year': i[3],
|
||||
'path': i[4],
|
||||
'position': i[5],
|
||||
'title': title,
|
||||
'author': author,
|
||||
'year': year,
|
||||
'path': path,
|
||||
'position': position,
|
||||
'isbn': i[6],
|
||||
'tags': i[7],
|
||||
'tags': tags,
|
||||
'hash': i[8]}
|
||||
|
||||
tooltip_string = title + '\nAuthor: ' + author + '\nYear: ' + str(year)
|
||||
@@ -64,19 +62,7 @@ class Library:
|
||||
if tags:
|
||||
search_workaround += tags
|
||||
|
||||
# Generate book state for passing onto the QStyledItemDelegate
|
||||
def generate_book_state(path, progress):
|
||||
if not os.path.exists(path):
|
||||
return 'deleted'
|
||||
|
||||
if progress:
|
||||
if progress == 'completed':
|
||||
return 'completed'
|
||||
else:
|
||||
return 'inprogress'
|
||||
else:
|
||||
return None
|
||||
state = generate_book_state(path, progress)
|
||||
file_exists = os.path.exists(path)
|
||||
|
||||
# Generate image pixmap and then pass it to the widget
|
||||
# as a QIcon
|
||||
@@ -95,7 +81,9 @@ class Library:
|
||||
item.setData(year, QtCore.Qt.UserRole + 2)
|
||||
item.setData(all_metadata, QtCore.Qt.UserRole + 3)
|
||||
item.setData(search_workaround, QtCore.Qt.UserRole + 4)
|
||||
item.setData(state, QtCore.Qt.UserRole + 5)
|
||||
item.setData(file_exists, QtCore.Qt.UserRole + 5)
|
||||
item.setData(i[8], QtCore.Qt.UserRole + 6) # File hash
|
||||
item.setData(position, QtCore.Qt.UserRole + 7)
|
||||
item.setIcon(QtGui.QIcon(img_pixmap))
|
||||
self.parent_window.viewModel.appendRow(item)
|
||||
|
||||
@@ -198,3 +186,9 @@ class Settings:
|
||||
current_profile3])
|
||||
self.settings.setValue('currentProfileIndex', current_profile_index)
|
||||
self.settings.endGroup()
|
||||
|
||||
|
||||
class MyAbsModel(QtGui.QStandardItemModel, QtCore.QAbstractItemModel):
|
||||
def __init__(self, parent=None):
|
||||
# We're using this to be able to access the match() method
|
||||
super(MyAbsModel, self).__init__(parent)
|
||||
|
66
widgets.py
66
widgets.py
@@ -1,7 +1,11 @@
|
||||
#!usr/bin/env python3
|
||||
|
||||
import os
|
||||
from PyQt5 import QtWidgets, QtGui, QtCore
|
||||
|
||||
import pie_chart
|
||||
|
||||
|
||||
class BookToolBar(QtWidgets.QToolBar):
|
||||
def __init__(self, parent=None):
|
||||
super(BookToolBar, self).__init__(parent)
|
||||
@@ -280,10 +284,15 @@ class Tab(QtWidgets.QWidget):
|
||||
|
||||
# TODO
|
||||
# Chapter position and vertical scrollbar position
|
||||
if not position:
|
||||
first_chapter_name = list(self.metadata['content'])[0]
|
||||
first_chapter_content = self.metadata['content'][first_chapter_name]
|
||||
self.contentView.setHtml(first_chapter_content)
|
||||
if position:
|
||||
current_chapter = position['current_chapter']
|
||||
else:
|
||||
self.generate_position()
|
||||
current_chapter = 1
|
||||
|
||||
chapter_name = list(self.metadata['content'])[current_chapter - 1]
|
||||
chapter_content = self.metadata['content'][chapter_name]
|
||||
self.contentView.setHtml(chapter_content)
|
||||
|
||||
self.gridLayout.addWidget(self.contentView, 0, 0, 1, 1)
|
||||
self.parent.addTab(self, title)
|
||||
@@ -293,6 +302,17 @@ class Tab(QtWidgets.QWidget):
|
||||
self.exit_fs.setContext(QtCore.Qt.ApplicationShortcut)
|
||||
self.exit_fs.activated.connect(self.exit_fullscreen)
|
||||
|
||||
def generate_position(self):
|
||||
total_chapters = len(self.metadata['content'].keys())
|
||||
# TODO
|
||||
# Calculate lines
|
||||
self.metadata['position'] = {
|
||||
'current_chapter': 1,
|
||||
'current_line': 0,
|
||||
'total_chapters': total_chapters,
|
||||
'read_lines': 0,
|
||||
'total_lines': 0}
|
||||
|
||||
def exit_fullscreen(self):
|
||||
self.contentView.setWindowFlags(QtCore.Qt.Widget)
|
||||
self.contentView.setWindowState(QtCore.Qt.WindowNoState)
|
||||
@@ -301,32 +321,48 @@ class Tab(QtWidgets.QWidget):
|
||||
|
||||
|
||||
class LibraryDelegate(QtWidgets.QStyledItemDelegate):
|
||||
def __init__(self, parent=None):
|
||||
def __init__(self, temp_dir, parent=None):
|
||||
super(LibraryDelegate, self).__init__(parent)
|
||||
self.temp_dir = temp_dir
|
||||
|
||||
def paint(self, painter, option, index):
|
||||
QtWidgets.QStyledItemDelegate.paint(self, painter, option, index)
|
||||
# This is a hint for the future
|
||||
# Color icon slightly red
|
||||
# if option.state & QtWidgets.QStyle.State_Selected:
|
||||
# painter.fillRect(option.rect, QtGui.QColor().fromRgb(255, 0, 0, 20))
|
||||
# Also, painter.setOpacity(n)
|
||||
|
||||
option = option.__class__(option)
|
||||
state = index.data(QtCore.Qt.UserRole + 5)
|
||||
if state:
|
||||
if state == 'deleted':
|
||||
painter.setOpacity(.5)
|
||||
file_exists = index.data(QtCore.Qt.UserRole + 5)
|
||||
position = index.data(QtCore.Qt.UserRole + 7)
|
||||
|
||||
# TODO
|
||||
# Calculate progress on the basis of lines
|
||||
|
||||
if position:
|
||||
current_chapter = position['current_chapter']
|
||||
total_chapters = position['total_chapters']
|
||||
progress_percent = int(current_chapter * 100 / total_chapters)
|
||||
|
||||
if not file_exists:
|
||||
read_icon = QtGui.QIcon.fromTheme('vcs-conflicting').pixmap(36)
|
||||
painter.setOpacity(.5)
|
||||
QtWidgets.QStyledItemDelegate.paint(self, painter, option, index)
|
||||
painter.setOpacity(1)
|
||||
if state == 'completed':
|
||||
elif current_chapter == total_chapters:
|
||||
QtWidgets.QStyledItemDelegate.paint(self, painter, option, index)
|
||||
read_icon = QtGui.QIcon.fromTheme('vcs-normal').pixmap(36)
|
||||
if state == 'inprogress':
|
||||
read_icon = QtGui.QIcon.fromTheme('vcs-locally-modified').pixmap(36)
|
||||
elif current_chapter == 1:
|
||||
QtWidgets.QStyledItemDelegate.paint(self, painter, option, index)
|
||||
else:
|
||||
QtWidgets.QStyledItemDelegate.paint(self, painter, option, index)
|
||||
pie_chart.GeneratePie(progress_percent, self.temp_dir).generate()
|
||||
svg_path = os.path.join(self.temp_dir, 'lector_progress.svg')
|
||||
read_icon = QtGui.QIcon(svg_path).pixmap(34)
|
||||
|
||||
x_draw = option.rect.bottomRight().x() - 30
|
||||
y_draw = option.rect.bottomRight().y() - 35
|
||||
painter.drawPixmap(x_draw, y_draw, read_icon)
|
||||
if current_chapter != 1:
|
||||
painter.drawPixmap(x_draw, y_draw, read_icon)
|
||||
|
||||
else:
|
||||
QtWidgets.QStyledItemDelegate.paint(self, painter, option, index)
|
||||
|
Reference in New Issue
Block a user