Compare commits
	
		
			19 Commits
		
	
	
		
			92ad7efaad
			...
			master
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 86ccee18fb | ||
|  | b2d67f7aea | ||
|  | 71b0a6a112 | ||
|  | 69fa955be1 | ||
|  | db4716b21b | ||
|  | 7fd369be74 | ||
|  | 30dd9b2bcd | ||
|  | a6316f27f6 | ||
|  | 6129258f1b | ||
|  | be07589f24 | ||
|  | f952879753 | ||
|  | e1d1946fa5 | ||
|  | 89486d7c97 | ||
|  | dfa604e846 | ||
|  | 7a02bbb262 | ||
|  | dd9f08aa5e | ||
|  | 49de6b1f35 | ||
|  | d46281fda7 | ||
|  | 9a60a5adb5 | 
							
								
								
									
										167
									
								
								MainWindow.py
									
									
									
									
									
								
							
							
						
						
									
										167
									
								
								MainWindow.py
									
									
									
									
									
								
							| @@ -1,12 +1,11 @@ | ||||
| import enum | ||||
| from typing import Any, cast | ||||
|  | ||||
| from PySide6.QtCore import ( | ||||
|     QByteArray, | ||||
|     QDate, | ||||
|     QModelIndex, | ||||
|     QPersistentModelIndex, | ||||
|     QPoint, | ||||
|     QSettings, | ||||
|     Qt, | ||||
|     Signal, | ||||
|     Slot, | ||||
| @@ -15,19 +14,30 @@ from PySide6.QtGui import ( | ||||
|     QCloseEvent, | ||||
|     QColor, | ||||
| ) | ||||
| from PySide6.QtSql import QSqlTableModel | ||||
| from PySide6.QtSql import QSqlQuery, QSqlTableModel | ||||
| from PySide6.QtWidgets import ( | ||||
|     QAbstractItemView, | ||||
|     QHBoxLayout, | ||||
|     QHeaderView, | ||||
|     QLabel, | ||||
|     QMainWindow, | ||||
|     QProgressBar, | ||||
|     QPushButton, | ||||
|     QSizePolicy, | ||||
|     QStyledItemDelegate, | ||||
|     QStyleOptionViewItem, | ||||
|     QWidget, | ||||
| ) | ||||
|  | ||||
| from docketModel import docketModel | ||||
| from lib.utils import QStyleOptionViewItemInit, readGeometry, writeGeometry | ||||
| from lib.utils import ( | ||||
|     QStyleOptionViewItemInit, | ||||
|     query_error, | ||||
|     readGeometry, | ||||
|     writeGeometry, | ||||
| ) | ||||
| from ui.MainWindow import Ui_MainWindow | ||||
| from workers import updateThread | ||||
| from workers import loadCases, updateThread | ||||
|  | ||||
|  | ||||
| class dateDelegate(QStyledItemDelegate): | ||||
| @@ -70,11 +80,19 @@ class activeDelegate(QStyledItemDelegate): | ||||
|  | ||||
|  | ||||
| class casesModel(QSqlTableModel): | ||||
|     class ColumnNames(enum.IntEnum): | ||||
|         case_id = 0 | ||||
|         docket_id = 1 | ||||
|         petitioners = 2 | ||||
|         respondents = 3 | ||||
|         date = 4 | ||||
|         active = 5 | ||||
|  | ||||
|     def flags(self, index: QModelIndex | QPersistentModelIndex) -> Qt.ItemFlag: | ||||
|         if not index.isValid(): | ||||
|             return Qt.ItemFlag.NoItemFlags | ||||
|         flags = super(casesModel, self).flags(index) | ||||
|         if index.column() == 5: | ||||
|         if index.column() == casesModel.ColumnNames.active: | ||||
|             flags = ( | ||||
|                 Qt.ItemFlag.ItemIsEnabled | ||||
|                 | Qt.ItemFlag.ItemIsEditable | ||||
| @@ -89,7 +107,7 @@ class casesModel(QSqlTableModel): | ||||
|     ) -> Any: | ||||
|         if not index.isValid(): | ||||
|             return None | ||||
|         if index.column() == 5: | ||||
|         if index.column() == casesModel.ColumnNames.active: | ||||
|             if role == Qt.ItemDataRole.CheckStateRole: | ||||
|                 value = super(casesModel, self).data(index) | ||||
|                 return ( | ||||
| @@ -107,7 +125,10 @@ class casesModel(QSqlTableModel): | ||||
|         value: Any, | ||||
|         role: int = Qt.ItemDataRole.DisplayRole, | ||||
|     ) -> bool: | ||||
|         if role == Qt.ItemDataRole.CheckStateRole and index.column() == 5: | ||||
|         if ( | ||||
|             role == Qt.ItemDataRole.CheckStateRole | ||||
|             and index.column() == casesModel.ColumnNames.active | ||||
|         ): | ||||
|             super(casesModel, self).setData(index, 1 if value else 0) | ||||
|             return True | ||||
|         return super().setData(index, value, role) | ||||
| @@ -115,6 +136,7 @@ class casesModel(QSqlTableModel): | ||||
|  | ||||
| class MainWindow(QMainWindow, Ui_MainWindow): | ||||
|     show_entries = Signal(int) | ||||
|     update_status = Signal() | ||||
|  | ||||
|     loadThread = None | ||||
|  | ||||
| @@ -129,11 +151,27 @@ class MainWindow(QMainWindow, Ui_MainWindow): | ||||
|             "1=1 ORDER BY SUBSTRING(docket_id, 1, 3), CAST(SUBSTRING(docket_id,4) AS INTEGER)" | ||||
|         ) | ||||
|         model.select() | ||||
|         model.setHeaderData(1, Qt.Orientation.Horizontal, "Docket") | ||||
|         model.setHeaderData(2, Qt.Orientation.Horizontal, "Petitioners") | ||||
|         model.setHeaderData(3, Qt.Orientation.Horizontal, "Respondents") | ||||
|         model.setHeaderData(4, Qt.Orientation.Horizontal, "Date") | ||||
|         model.setHeaderData(5, Qt.Orientation.Horizontal, "Active") | ||||
|         model.setHeaderData( | ||||
|             casesModel.ColumnNames.docket_id, | ||||
|             Qt.Orientation.Horizontal, | ||||
|             "Docket", | ||||
|         ) | ||||
|         model.setHeaderData( | ||||
|             casesModel.ColumnNames.petitioners, | ||||
|             Qt.Orientation.Horizontal, | ||||
|             "Petitioners", | ||||
|         ) | ||||
|         model.setHeaderData( | ||||
|             casesModel.ColumnNames.respondents, | ||||
|             Qt.Orientation.Horizontal, | ||||
|             "Respondents", | ||||
|         ) | ||||
|         model.setHeaderData( | ||||
|             casesModel.ColumnNames.date, Qt.Orientation.Horizontal, "Date" | ||||
|         ) | ||||
|         model.setHeaderData( | ||||
|             casesModel.ColumnNames.active, Qt.Orientation.Horizontal, "Active" | ||||
|         ) | ||||
|         self.casesView.setModel(model) | ||||
|  | ||||
|         self.casesView.setSelectionMode( | ||||
| @@ -142,23 +180,33 @@ class MainWindow(QMainWindow, Ui_MainWindow): | ||||
|         self.casesView.setSelectionBehavior( | ||||
|             QAbstractItemView.SelectionBehavior.SelectRows | ||||
|         ) | ||||
|         self.casesView.hideColumn(0) | ||||
|         self.casesView.hideColumn(casesModel.ColumnNames.case_id) | ||||
|         self.casesView.setItemDelegate(activeDelegate()) | ||||
|         self.casesView.setItemDelegateForColumn(4, dateDelegate()) | ||||
|         self.casesView.resizeColumnToContents(1) | ||||
|         self.casesView.resizeColumnToContents(4) | ||||
|         self.casesView.setItemDelegateForColumn( | ||||
|             casesModel.ColumnNames.date, dateDelegate() | ||||
|         ) | ||||
|         self.casesView.resizeColumnToContents(casesModel.ColumnNames.docket_id) | ||||
|         self.casesView.resizeColumnToContents(casesModel.ColumnNames.date) | ||||
|         header = self.casesView.horizontalHeader() | ||||
|         header.setSectionResizeMode(2, QHeaderView.ResizeMode.Fixed) | ||||
|         header.setSectionResizeMode(3, QHeaderView.ResizeMode.Fixed) | ||||
|         header.setSectionResizeMode( | ||||
|             casesModel.ColumnNames.petitioners, QHeaderView.ResizeMode.Fixed | ||||
|         ) | ||||
|         header.setSectionResizeMode( | ||||
|             casesModel.ColumnNames.respondents, QHeaderView.ResizeMode.Fixed | ||||
|         ) | ||||
|         self.show() | ||||
|         remaining = ( | ||||
|             self.casesView.width() | ||||
|             - header.sectionSize(1) | ||||
|             - header.sectionSize(4) | ||||
|             - header.sectionSize(casesModel.ColumnNames.docket_id) | ||||
|             - header.sectionSize(casesModel.ColumnNames.date) | ||||
|             - 5 | ||||
|         ) | ||||
|         self.casesView.setColumnWidth(2, int(remaining * 0.5)) | ||||
|         self.casesView.setColumnWidth(3, int(remaining * 0.5)) | ||||
|         self.casesView.setColumnWidth( | ||||
|             casesModel.ColumnNames.petitioners, int(remaining * 0.5) | ||||
|         ) | ||||
|         self.casesView.setColumnWidth( | ||||
|             casesModel.ColumnNames.respondents, int(remaining * 0.5) | ||||
|         ) | ||||
|         self.casesView.verticalHeader().hide() | ||||
|         self.casesView.resizeRowsToContents() | ||||
|         self.casesView.doubleClicked.connect(self.rowClicked) | ||||
| @@ -169,12 +217,85 @@ class MainWindow(QMainWindow, Ui_MainWindow): | ||||
|         self.docketView.horizontalHeader().setSectionResizeMode( | ||||
|             1, QHeaderView.ResizeMode.Stretch | ||||
|         ) | ||||
|  | ||||
|         widget = QWidget() | ||||
|         layout = QHBoxLayout(widget) | ||||
|         self.status = QLabel("Status") | ||||
|         layout.addWidget(self.status) | ||||
|         layout.addStretch() | ||||
|         self.progress = QProgressBar() | ||||
|         self.progress.setRange(0, 100) | ||||
|         self.progress.setValue(0) | ||||
|         self.progress.setMinimumWidth(150) | ||||
|         self.progress.setMaximumWidth(150) | ||||
|         self.progress.setSizePolicy( | ||||
|             QSizePolicy.Policy.Fixed, QSizePolicy.Policy.Preferred | ||||
|         ) | ||||
|         layout.addWidget(self.progress) | ||||
|         self.loadButton = QPushButton() | ||||
|         self.loadButton.setObjectName("loadButton") | ||||
|         self.loadButton.setText("Load Cases") | ||||
|         layout.addWidget(self.loadButton) | ||||
|         self.statusbar.addWidget(widget, 10) | ||||
|  | ||||
|         self.loadButton.clicked.connect(self.loadCases) | ||||
|         self.update_status.connect(self.statusBarUpdate) | ||||
|         self.update_status.emit() | ||||
|         self.docketLabel.setText("") | ||||
|         return | ||||
|  | ||||
|     def closeEvent(self, event: QCloseEvent) -> None: | ||||
|         writeGeometry(self) | ||||
|         super().closeEvent(event) | ||||
|         return | ||||
|  | ||||
|     @Slot() | ||||
|     def loadCases(self) -> None: | ||||
|         if self.loadThread is None: | ||||
|             self.loadThread = loadCases() | ||||
|             self.loadThread.caseLoaded.connect(self.progress.setValue) | ||||
|             self.loadThread.finished.connect(self.loadCasesDone) | ||||
|         self.loadButton.setEnabled(False) | ||||
|         self.loadThread.start() | ||||
|         return | ||||
|  | ||||
|     @Slot() | ||||
|     def loadCasesDone(self) -> None: | ||||
|         self.loadButton.setEnabled(True) | ||||
|         self.update_status.emit() | ||||
|         return | ||||
|  | ||||
|     @Slot() | ||||
|     def statusBarUpdate(self) -> None: | ||||
|         if self.loadThread is None: | ||||
|             year = "" | ||||
|             max = "unknown" | ||||
|         else: | ||||
|             year = self.loadThread.year | ||||
|             max = str(self.loadThread.number) | ||||
|         query = QSqlQuery() | ||||
|         query.prepare( | ||||
|             "SELECT COUNT(*) AS number_active FROM cases WHERE active=1" | ||||
|         ) | ||||
|         if not query.exec(): | ||||
|             query_error(query) | ||||
|         if query.next(): | ||||
|             active = query.value("number_active") | ||||
|         query.prepare( | ||||
|             "select SUBSTRING(docket_id,1,2) AS year, " | ||||
|             "SUBSTRING(docket_id,3,1) AS type, " | ||||
|             "MAX(CAST(SUBSTRING(docket_id,4) AS INTEGER)) AS number " | ||||
|             "FROM cases " | ||||
|             "GROUP BY year, type " | ||||
|             "ORDER BY type, year, number" | ||||
|             "LIMIT 2" | ||||
|         ) | ||||
|         # if not query.exec(): | ||||
|         #    query_error(query) | ||||
|         msg = f"Oldest: {year}, Active: {active}, Max. Case: {max}" | ||||
|         self.status.setText(msg) | ||||
|         return | ||||
|  | ||||
|     @Slot(QModelIndex)  # type: ignore | ||||
|     def rowClicked(self, index: QModelIndex) -> None: | ||||
|         if not index.isValid(): | ||||
|   | ||||
| @@ -1,10 +1,9 @@ | ||||
| import datetime | ||||
| from typing import Any, cast | ||||
| import enum | ||||
| from typing import Any | ||||
|  | ||||
| from PySide6.QtCore import ( | ||||
|     QAbstractTableModel, | ||||
|     QDate, | ||||
|     QDateTime, | ||||
|     QModelIndex, | ||||
|     QPersistentModelIndex, | ||||
|     Qt, | ||||
| @@ -18,6 +17,10 @@ from lib.utils import query_error | ||||
| class docketModel(QAbstractTableModel): | ||||
|     entries: list[list[str | None]] = [] | ||||
|  | ||||
|     class ColumnNames(enum.IntEnum): | ||||
|         date = 0 | ||||
|         text = 1 | ||||
|  | ||||
|     def __init__(self, case_id: int | None = None) -> None: | ||||
|         super(docketModel, self).__init__() | ||||
|         if case_id == None: | ||||
|   | ||||
| @@ -1,13 +1,13 @@ | ||||
| from typing import cast | ||||
|  | ||||
| from PySide6.QtCore import ( | ||||
|     QAbstractItemModel, | ||||
|     QDir, | ||||
|     QFile, | ||||
|     QModelIndex, | ||||
|     QObject, | ||||
|     QPersistentModelIndex, | ||||
|     QPoint, | ||||
|     QPointF, | ||||
|     QRect, | ||||
|     QSize, | ||||
|     Qt, | ||||
|     QUrl, | ||||
| @@ -16,7 +16,7 @@ from PySide6.QtCore import ( | ||||
| ) | ||||
| from PySide6.QtGui import ( | ||||
|     QMouseEvent, | ||||
|     QPalette, | ||||
|     QPainter, | ||||
|     QTextDocument, | ||||
| ) | ||||
| from PySide6.QtNetwork import ( | ||||
| @@ -26,7 +26,7 @@ from PySide6.QtNetwork import ( | ||||
| ) | ||||
| from PySide6.QtWidgets import ( | ||||
|     QAbstractItemView, | ||||
|     QSizePolicy, | ||||
|     QStyle, | ||||
|     QStyledItemDelegate, | ||||
|     QStyleOptionViewItem, | ||||
|     QTableView, | ||||
| @@ -34,6 +34,7 @@ from PySide6.QtWidgets import ( | ||||
|     QWidget, | ||||
| ) | ||||
|  | ||||
| from lib.utils import QStyleOptionViewItemInit | ||||
| from pdfView import PDFViewer | ||||
|  | ||||
|  | ||||
| @@ -47,43 +48,43 @@ class docketEntryDelegate(QStyledItemDelegate): | ||||
|  | ||||
|     def sizeHint( | ||||
|         self, | ||||
|         _: QStyleOptionViewItem, | ||||
|         option: QStyleOptionViewItem, | ||||
|         index: QModelIndex | QPersistentModelIndex, | ||||
|     ) -> QSize: | ||||
|         widget = self.view.indexWidget(index) | ||||
|         return widget.sizeHint() | ||||
|  | ||||
|  | ||||
| class docketEntry(QTextEdit): | ||||
|     def __init__(self, parent: QWidget | None = None) -> None: | ||||
|         super(docketEntry, self).__init__(parent) | ||||
|         self.setSizePolicy( | ||||
|             QSizePolicy.Policy.Preferred, QSizePolicy.Policy.Fixed | ||||
|         ) | ||||
|         return | ||||
|  | ||||
|     def mousePressEvent(self, e: QMouseEvent) -> None: | ||||
|         super().mousePressEvent(e) | ||||
|         anchor = self.anchorAt(e.pos()) | ||||
|         if anchor: | ||||
|             obj = cast(QObject, self) | ||||
|             while not isinstance(obj, docketTableView) and obj is not None: | ||||
|                 obj = obj.parent() | ||||
|             assert obj is not None | ||||
|             index = obj.indexAt(obj.mapFromGlobal(self.mapToGlobal(e.pos()))) | ||||
|             obj.anchorSignal.emit(index, anchor) | ||||
|         return | ||||
|  | ||||
|     def sizeHint(self) -> QSize: | ||||
|         size = self.size() | ||||
|         options = cast(QStyleOptionViewItemInit, option) | ||||
|         self.initStyleOption(options, index) | ||||
|         doc = QTextDocument() | ||||
|         doc.setPlainText(self.document().toPlainText()) | ||||
|         doc.setTextWidth(size.width()) | ||||
|         docSize = doc.size() | ||||
|         return QSize(size.width(), int(docSize.height())) | ||||
|         doc.setHtml(options.text) | ||||
|         doc.setTextWidth(options.rect.width()) | ||||
|         return QSize(int(doc.idealWidth()), int(doc.size().height())) | ||||
|  | ||||
|     def paint( | ||||
|         self, | ||||
|         painter: QPainter, | ||||
|         option: QStyleOptionViewItem, | ||||
|         index: QModelIndex | QPersistentModelIndex, | ||||
|     ) -> None: | ||||
|         options = cast(QStyleOptionViewItemInit, option) | ||||
|         self.initStyleOption(options, index) | ||||
|         painter.save() | ||||
|  | ||||
|         doc = QTextDocument() | ||||
|         doc.setHtml(options.text) | ||||
|         doc.setTextWidth(options.rect.width()) | ||||
|  | ||||
|         options.text = "" | ||||
|         options.widget.style().drawControl( | ||||
|             QStyle.ControlElement.CE_ItemViewItem, options, painter | ||||
|         ) | ||||
|         painter.translate(options.rect.left(), options.rect.top()) | ||||
|         clip = QRect(0, 0, options.rect.width(), options.rect.height()) | ||||
|         doc.drawContents(painter, clip) | ||||
|         painter.restore() | ||||
|         return | ||||
|  | ||||
|  | ||||
| class docketTableView(QTableView): | ||||
|  | ||||
|     manager: QNetworkAccessManager | ||||
|     clickedEvent = Signal(QPoint) | ||||
|     anchorSignal = Signal(QModelIndex, str) | ||||
| @@ -97,12 +98,6 @@ class docketTableView(QTableView): | ||||
|         self.manager.finished.connect(self.getDone) | ||||
|         return | ||||
|  | ||||
|     def setModel(self, model: QAbstractItemModel | None) -> None: | ||||
|         assert model is not None | ||||
|         super().setModel(model) | ||||
|         self.model().modelReset.connect(self.modelReset) | ||||
|         return | ||||
|  | ||||
|     @Slot(QNetworkReply)  # type: ignore | ||||
|     def getDone(self, reply: QNetworkReply) -> None: | ||||
|         dest = QFile("." + reply.url().path()) | ||||
| @@ -134,16 +129,36 @@ class docketTableView(QTableView): | ||||
|             self.manager.get(QNetworkRequest(url)) | ||||
|         return | ||||
|  | ||||
|     def modelReset(self) -> None: | ||||
|         model = self.model() | ||||
|         red = QPalette() | ||||
|         red.setColor(QPalette.ColorRole.Base, Qt.GlobalColor.red) | ||||
|         for row in range(0, model.rowCount()): | ||||
|             index = model.index(row, 1) | ||||
|             widget = docketEntry() | ||||
|             widget.setHtml(model.data(index, Qt.ItemDataRole.DisplayRole)) | ||||
|             widget.setAutoFillBackground(False) | ||||
|             widget.setReadOnly(True) | ||||
|             widget.setPalette(red) | ||||
|             self.setIndexWidget(index, widget) | ||||
|     def mousePressEvent(self, event: QMouseEvent) -> None: | ||||
|         # | ||||
|         # The mouse has been pressed somewere in our rect.  We need to translate that to a click | ||||
|         # within the cell with the document.  This will allow the document to find anchors. | ||||
|         # | ||||
|         index = self.indexAt(event.pos()) | ||||
|         if not index.isValid(): | ||||
|             return | ||||
|         if index.column() != 1: | ||||
|             return | ||||
|         doc = QTextDocument() | ||||
|         doc.setHtml(index.data(Qt.ItemDataRole.DisplayRole)) | ||||
|         doc.setTextWidth(self.columnWidth(index.column())) | ||||
|         # | ||||
|         # We need to map our click position to a position within the cell. | ||||
|         # | ||||
|         pos = event.position() | ||||
|         new_pos = QPointF( | ||||
|             pos.x() - self.horizontalScrollBar().value(), | ||||
|             pos.y() - self.verticalScrollBar().value(), | ||||
|         ) | ||||
|         rowHeight = 0 | ||||
|         for row in range(0, index.row()): | ||||
|             rowHeight += self.rowHeight(row) | ||||
|         cell_pos = QPointF( | ||||
|             new_pos.x() - self.columnWidth(0), new_pos.y() - rowHeight | ||||
|         ) | ||||
|         te = QTextEdit() | ||||
|         te.setDocument(doc) | ||||
|         anchor = te.anchorAt(cell_pos.toPoint()) | ||||
|         if anchor: | ||||
|             self.anchorSignal.emit(index, anchor) | ||||
|         return | ||||
|   | ||||
| @@ -1,5 +1,4 @@ | ||||
| from datetime import date, datetime | ||||
| from decimal import Decimal | ||||
| from datetime import date | ||||
|  | ||||
| from pony.orm import (  # type: ignore[import-untyped] | ||||
|     Database, | ||||
| @@ -10,8 +9,7 @@ from pony.orm import (  # type: ignore[import-untyped] | ||||
|     Set, | ||||
|     set_sql_debug, | ||||
| ) | ||||
|  | ||||
| db = Database | ||||
| from PySide6.QtCore import QSettings | ||||
|  | ||||
| db = Database() | ||||
|  | ||||
| @@ -50,13 +48,18 @@ class History(db.Entity):  # type: ignore[name-defined] | ||||
|     number = Required(int) | ||||
|  | ||||
|  | ||||
| set_sql_debug(True) | ||||
| db.bind( | ||||
|     provider="mysql", | ||||
|     user="scotus", | ||||
|     host="ceph5", | ||||
|     database="scotus", | ||||
|     password="lechOtvirf8Om/", | ||||
| ) | ||||
| db.generate_mapping(create_tables=True) | ||||
| db.disconnect() | ||||
| def updateDatabase(settings: QSettings) -> None: | ||||
|     set_sql_debug(True) | ||||
|     engine = settings.value("engine") | ||||
|     if engine not in ["QMYSQL", "MARIADB"]: | ||||
|         raise Exception(f"Unknown database engine: {engine}") | ||||
|     db.bind( | ||||
|         provider="mysql", | ||||
|         user=settings.value("user"), | ||||
|         host=settings.value("hostname"), | ||||
|         database=settings.value("databasename"), | ||||
|         password=settings.value("password"), | ||||
|     ) | ||||
|     db.generate_mapping(create_tables=True) | ||||
|     db.disconnect() | ||||
|     return | ||||
|   | ||||
							
								
								
									
										20
									
								
								lib/utils.py
									
									
									
									
									
								
							
							
						
						
									
										20
									
								
								lib/utils.py
									
									
									
									
									
								
							| @@ -1,6 +1,11 @@ | ||||
| from typing import NoReturn, cast | ||||
| from typing import NoReturn | ||||
|  | ||||
| from PySide6.QtCore import QByteArray, QCoreApplication, QObject, QRect, QSettings, Signal | ||||
| from PySide6.QtCore import ( | ||||
|     QByteArray, | ||||
|     QCoreApplication, | ||||
|     QRect, | ||||
|     QSettings, | ||||
| ) | ||||
| from PySide6.QtGui import QColor, QFont | ||||
| from PySide6.QtSql import QSqlQuery | ||||
| from PySide6.QtWidgets import QStyleOptionViewItem, QWidget | ||||
| @@ -29,23 +34,26 @@ class QStyleOptionViewItemInit(QStyleOptionViewItem): | ||||
|     font: QFont | ||||
|     text: str | ||||
|  | ||||
| def openSettings(group:str|None = None) -> QSettings: | ||||
|  | ||||
| def openSettings(group: str | None = None) -> QSettings: | ||||
|     settings = QSettings("Troglodite Services", "SCOTUS Watch") | ||||
|     if group is not None: | ||||
|         settings.beginGroup(group) | ||||
|     return settings | ||||
|  | ||||
| def readGeometry(widget:QWidget) -> None: | ||||
|  | ||||
| def readGeometry(widget: QWidget) -> None: | ||||
|     settings = openSettings(widget.objectName()) | ||||
|     geometry = settings.value('geometry', QByteArray()) | ||||
|     geometry = settings.value("geometry", QByteArray()) | ||||
|     assert isinstance(geometry, QByteArray) | ||||
|     if not geometry.isEmpty(): | ||||
|         widget.restoreGeometry(geometry) | ||||
|     settings.endGroup() | ||||
|     return | ||||
|  | ||||
|  | ||||
| def writeGeometry(widget: QWidget) -> None: | ||||
|     settings = openSettings(widget.objectName()) | ||||
|     settings.setValue('geometry', widget.saveGeometry()) | ||||
|     settings.setValue("geometry", widget.saveGeometry()) | ||||
|     settings.endGroup() | ||||
|     return | ||||
|   | ||||
							
								
								
									
										21
									
								
								pdfView.py
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								pdfView.py
									
									
									
									
									
								
							| @@ -1,7 +1,7 @@ | ||||
| import math | ||||
|  | ||||
| from PySide6.QtCore import QFile, QModelIndex, QPoint, Signal, Slot | ||||
| from PySide6.QtGui import QCloseEvent | ||||
| from PySide6.QtCore import QFile, QModelIndex, QPoint, QSize, Signal, Slot | ||||
| from PySide6.QtGui import QCloseEvent, QGuiApplication | ||||
| from PySide6.QtPdf import QPdfBookmarkModel, QPdfDocument | ||||
| from PySide6.QtPdfWidgets import QPdfView | ||||
| from PySide6.QtWidgets import QComboBox, QDialog, QMenuBar, QSpinBox, QWidget | ||||
| @@ -83,7 +83,7 @@ class PDFViewer(QDialog, Ui_pdfViewer): | ||||
|     def __init__(self, parent: QWidget | None) -> None: | ||||
|         super(PDFViewer, self).__init__(parent) | ||||
|         self.setupUi(self) | ||||
|         self.setObjectName('PDFViewer') | ||||
|         self.setObjectName("PDFViewer") | ||||
|         readGeometry(self) | ||||
|         print(self.objectName()) | ||||
|         self.zoomSelector = ZoomSelector(self) | ||||
| @@ -92,7 +92,7 @@ class PDFViewer(QDialog, Ui_pdfViewer): | ||||
|  | ||||
|         self.zoomSelector.setMaximumWidth(150) | ||||
|         self.mainToolBar.insertWidget(self.actionZoom_In, self.zoomSelector) | ||||
|         self.mainToolBar.insertWidget(self.actionForward, self.pageSelector) | ||||
|         self.mainToolBar.insertWidget(self.actionNext_Page, self.pageSelector) | ||||
|         self.pageSelector.valueChanged.connect(self.page_selected) | ||||
|         nav = self.pdfView.pageNavigator() | ||||
|         nav.currentPageChanged.connect(self.pageSelector.setValue) | ||||
| @@ -118,16 +118,13 @@ class PDFViewer(QDialog, Ui_pdfViewer): | ||||
|         return | ||||
|  | ||||
|     def closeEvent(self, event: QCloseEvent) -> None: | ||||
|         print("closeEvent") | ||||
|         writeGeometry(self) | ||||
|         super().closeEvent(event) | ||||
|         return | ||||
|  | ||||
|     @Slot(QFile)  # type: ignore | ||||
|     def open(self, file: QFile) -> None: | ||||
|         # assert file.exists() | ||||
|         # if not file.isOpen(): | ||||
|         # file.open(QFile.OpenModeFlag.ReadOnly) | ||||
|         # self.document.load(file) | ||||
|         self.document.load(file.fileName()) | ||||
|         document_title = self.document.metaData( | ||||
|             QPdfDocument.MetaDataField.Title | ||||
| @@ -135,6 +132,14 @@ class PDFViewer(QDialog, Ui_pdfViewer): | ||||
|         self.setWindowTitle(document_title if document_title else "PDF Viewer") | ||||
|         self.page_selected(0) | ||||
|         self.pageSelector.setMaximum(self.document.pageCount() - 1) | ||||
|         if self.zoomSelector.currentIndex() == 8: | ||||
|             pageSize = self.document.pagePointSize(0) | ||||
|             dpi = QGuiApplication.primaryScreen().physicalDotsPerInch() | ||||
|             size = QSize( | ||||
|                 int(pageSize.width() / 72 * dpi), self.pdfView.height() | ||||
|             ) | ||||
|             self.pdfView.resize(size) | ||||
|             self.splitter.setSizes([100, size.width()]) | ||||
|         return | ||||
|  | ||||
|     @Slot(QModelIndex)  # type: ignore | ||||
|   | ||||
| @@ -1,16 +1,21 @@ | ||||
| #!venv/bin/python3 | ||||
| import faulthandler | ||||
| import sys | ||||
| from typing import cast | ||||
|  | ||||
| from PySide6.QtCore import QSettings | ||||
| from PySide6.QtCore import QFile | ||||
| from PySide6.QtSql import ( | ||||
|     QSqlDatabase, | ||||
| ) | ||||
| from PySide6.QtWidgets import ( | ||||
|     QApplication, | ||||
|     QDialog, | ||||
|     QMainWindow, | ||||
| ) | ||||
|  | ||||
| from lib.dbConfig import updateDatabase | ||||
| from lib.utils import openSettings | ||||
| from setupDialog import setupDialog | ||||
|  | ||||
| faulthandler.enable() | ||||
|  | ||||
| @@ -20,17 +25,27 @@ from MainWindow import MainWindow | ||||
| def main() -> int: | ||||
|     # | ||||
|     app = QApplication(sys.argv) | ||||
|     settings = openSettings('database') | ||||
|     settings = openSettings("database") | ||||
|     file = QFile(settings.fileName()) | ||||
|     if not file.exists(): | ||||
|         result = setupDialog().exec() | ||||
|         if result == QDialog.DialogCode.Rejected: | ||||
|             print(f"We require database credentials") | ||||
|             return 2 | ||||
|     updateDatabase(settings) | ||||
|     db = QSqlDatabase.addDatabase("QMYSQL") | ||||
|     db.setHostName(settings.value("hostname")) | ||||
|     db.setDatabaseName(settings.value("databasename","scotus")) # type: ignore | ||||
|     db.setUserName(settings.value("user","scotus")) # type: ignore | ||||
|     port = settings.value("port", "3306") | ||||
|     assert isinstance(port, str) | ||||
|     db.setPort(int(port)) | ||||
|     db.setDatabaseName(settings.value("databasename", "scotus"))  # type: ignore | ||||
|     db.setUserName(settings.value("user", "scotus"))  # type: ignore | ||||
|     db.setPassword(settings.value("password")) | ||||
|     settings.endGroup() | ||||
|     db.open() | ||||
|     import lib.dbConfig | ||||
|  | ||||
|     window = MainWindow() | ||||
|     assert isinstance(window, QMainWindow) | ||||
|     return app.exec() | ||||
|  | ||||
|  | ||||
|   | ||||
							
								
								
									
										87
									
								
								scotusPdfView.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								scotusPdfView.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | ||||
| from PySide6.QtCore import QEvent, QObject, QPointF, QRect, QRectF, Qt | ||||
| from PySide6.QtGui import QColor, QMouseEvent, QPaintEvent, QPainter, QPen | ||||
| from PySide6.QtPdfWidgets import QPdfView | ||||
| from PySide6.QtWidgets import QApplication, QWidget | ||||
|  | ||||
| def printParentHierarchy(obj: QObject, indent: int = 0): | ||||
|     if obj: | ||||
|         indentStr = '  ' * indent * 2 | ||||
|         print(f"{indentStr}{obj.metaObject().className()}:{obj.objectName()}") | ||||
|         printParentHierarchy(obj.parent(), indent + 1) | ||||
|     return | ||||
|          | ||||
| class scotusPdfView(QPdfView): | ||||
|  | ||||
|     drawing: bool | ||||
|     start_point: QPointF | ||||
|     end_point: QPointF | ||||
|      | ||||
|     def formatRect(self, rect: QRectF|QRect) -> str: | ||||
|         return f"({rect.left()},{rect.top()}):({rect.right()},{rect.bottom()})" | ||||
|      | ||||
|     def __init__(self, parent: QWidget) -> None: | ||||
|         self.drawing = False | ||||
|         super(scotusPdfView, self).__init__(parent) | ||||
|         print(self.viewport()) | ||||
|         return | ||||
|  | ||||
|     def paintEvent(self, event: QPaintEvent) -> None: | ||||
|         super(scotusPdfView,self).paintEvent(event) | ||||
|         if self.drawing: | ||||
|             page = self.pageNavigator().currentPage() | ||||
|             viewport = self.viewport() | ||||
|             painter = QPainter(viewport) | ||||
|             doc = self.document() | ||||
|             dpi = QApplication.primaryScreen().logicalDotsPerInch() | ||||
|             # | ||||
|             # XXX: Our mouse events are storing coordinates in the "self" | ||||
|             # coordinate system.  We need them in the viewport system | ||||
|             # | ||||
|  | ||||
|             rect = QRectF(viewport.mapFrom(self, self.start_point), | ||||
|                           viewport.mapFrom(self, self.end_point)) | ||||
|             redSolid = QPen(Qt.GlobalColor.red, 2, Qt.PenStyle.SolidLine) | ||||
|             blueDash = QPen(Qt.GlobalColor.blue, 2, Qt.PenStyle.DashLine) | ||||
|             greenDash = QPen(Qt.GlobalColor.green, 2, Qt.PenStyle.DashLine) | ||||
|             painter.setPen(redSolid) | ||||
|             painter.drawRect(rect) # Mouse drag box | ||||
|  | ||||
|             selectRect = QRectF(rect.left()/dpi*72, | ||||
|                                 rect.top()/dpi*72, | ||||
|                                 rect.right()/dpi*72, | ||||
|                                 rect.bottom()/dpi*72) | ||||
|             painter.setPen(blueDash) | ||||
|             selection = doc.getSelectionAtIndex(page, 10, 50) | ||||
|             selection = doc.getSelection(page, selectRect.topLeft(), selectRect.bottomRight()) | ||||
|             bb = selection.boundingRectangle() | ||||
|             painter.drawRect(bb) # Selection bounding box | ||||
|             print(selection.text()) | ||||
|             print() | ||||
|  | ||||
|             painter.setPen(greenDash) | ||||
|             bb.setLeft(bb.left()/72*dpi) | ||||
|             bb.setTop(bb.top()/72*dpi) | ||||
|             bb.setRight(bb.right()/72*dpi) | ||||
|             bb.setBottom(bb.bottom()/72*dpi) | ||||
|             painter.drawRect(bb) # Selection translated into viewport coordinates? | ||||
|         return | ||||
|  | ||||
|     def mousePressEvent(self, event: QMouseEvent) -> None: | ||||
|         if event.button() == Qt.MouseButton.LeftButton: | ||||
|             self.start_point = event.pos() | ||||
|             self.drawing = True | ||||
|         return | ||||
|      | ||||
|     def mouseMoveEvent(self, event: QMouseEvent) -> None: | ||||
|         if self.drawing: | ||||
|             pos = event.pos() | ||||
|             self.end_point = pos | ||||
|             viewport = self.viewport() | ||||
|             viewport.update() | ||||
|         return | ||||
|  | ||||
|     def mouseReleaseEvent(self, event: QMouseEvent) -> None: | ||||
|         if event.button() == Qt.MouseButton.LeftButton: | ||||
|             self.drawing = False | ||||
|             self.viewport().update() | ||||
|         return | ||||
							
								
								
									
										68
									
								
								setupDialog.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								setupDialog.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,68 @@ | ||||
| from PySide6.QtCore import Qt, Slot | ||||
| from PySide6.QtGui import QPalette | ||||
| from PySide6.QtWidgets import QDialog, QStatusBar, QVBoxLayout, QWidget | ||||
|  | ||||
| from lib.utils import openSettings | ||||
| from ui.dbSetup import Ui_setupDialog | ||||
|  | ||||
|  | ||||
| class setupDialog(QDialog, Ui_setupDialog): | ||||
|     def __init__(self, parent: QWidget | None = None) -> None: | ||||
|         super(setupDialog, self).__init__(parent) | ||||
|         self.setupUi(self) | ||||
|         self.sqliteEdit.setEnabled(False) | ||||
|  | ||||
|         layout = self.layout() | ||||
|         assert isinstance(layout, QVBoxLayout) | ||||
|         self.statusBar = QStatusBar(self) | ||||
|         self.statusBar.setObjectName("statusBar") | ||||
|         where = layout.indexOf(self.buttonBox) | ||||
|         layout.insertWidget(where, self.statusBar) | ||||
|         self.pwEdit.editingFinished.connect(self.pwDone) | ||||
|         self.pwConfirmEdit.editingFinished.connect(self.pwDone) | ||||
|         return | ||||
|  | ||||
|     @Slot() | ||||
|     def pwDone(self) -> None: | ||||
|         pw = self.pwEdit.text() | ||||
|         confirm = self.pwConfirmEdit.text() | ||||
|  | ||||
|         if pw != confirm: | ||||
|             palette = self.statusBar.palette() | ||||
|             palette.setColor(QPalette.ColorRole.WindowText, Qt.GlobalColor.red) | ||||
|             self.statusBar.setPalette(palette) | ||||
|             self.statusBar.showMessage("Password doesn't match") | ||||
|             self.pwGood = False | ||||
|         else: | ||||
|             self.statusBar.clearMessage() | ||||
|             self.pwGood = True | ||||
|         return | ||||
|  | ||||
|     @Slot(int)  # type: ignore | ||||
|     def done(self, r: int) -> None: | ||||
|         if r == QDialog.DialogCode.Rejected: | ||||
|             super(setupDialog, self).done(r) | ||||
|             return | ||||
|         self.pwDone() | ||||
|         if not self.pwGood: | ||||
|             return | ||||
|         super(setupDialog, self).done(r) | ||||
|         return | ||||
|  | ||||
|     @Slot() | ||||
|     def accept(self) -> None: | ||||
|         settings = openSettings("database") | ||||
|         settings.setValue("hostname", self.hostEdit.text()) | ||||
|         settings.setValue("databasename", self.dbEdit.text()) | ||||
|         settings.setValue("user", self.userEdit.text()) | ||||
|         settings.setValue("password", self.pwEdit.text()) | ||||
|         settings.setValue("port", self.portEdit.text()) | ||||
|         if self.comboBox.currentIndex() == 0: | ||||
|             settings.setValue("engine", "QMYSQL") | ||||
|         elif self.comboBox.currentIndex() == 1: | ||||
|             settings.setValue("engine", "QSQLITE") | ||||
|         # Other types: QDB2, QIBASE, QOCI, QODBC, QPSQL, QMIMER | ||||
|         else: | ||||
|             print("Bad Database Type") | ||||
|         settings.sync() | ||||
|         return super().accept() | ||||
							
								
								
									
										14
									
								
								ui/Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								ui/Makefile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| depfiles := $(shell awk '/file/{gsub(/<\/?file>/,"");print}' resources.qrc) | ||||
| SRC := $(wildcard *.ui) | ||||
| PY := $(patsubst %.ui, %.py, ${SRC}) | ||||
| all: resources.rcc ${PY} | ||||
| %.py: %.ui | ||||
| 	pyside6-uic $< >$@ | ||||
| resources.rcc: resources.qrc | ||||
| 	pyside6-rcc --binary resources.qrc -o resources.rcc | ||||
| resources.qrc: ${depfiles} | ||||
| 	touch resources.qrc | ||||
| %.qm:%.ts | ||||
| 	pyside6-lrelease $< | ||||
| %ts: *.ui ../main.py ../lib/*.py | ||||
| 	pyside6-lupdate *.ui ../main.py ../lib/*.py -ts $@ | ||||
							
								
								
									
										176
									
								
								ui/dbSetup.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										176
									
								
								ui/dbSetup.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,176 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
|  | ||||
| ################################################################################ | ||||
| ## Form generated from reading UI file 'dbSetup.ui' | ||||
| ## | ||||
| ## Created by: Qt User Interface Compiler version 6.8.2 | ||||
| ## | ||||
| ## WARNING! All changes made in this file will be lost when recompiling UI file! | ||||
| ################################################################################ | ||||
|  | ||||
| from PySide6.QtCore import (QCoreApplication, QDate, QDateTime, QLocale, | ||||
|     QMetaObject, QObject, QPoint, QRect, | ||||
|     QSize, QTime, QUrl, Qt) | ||||
| from PySide6.QtGui import (QAction, QBrush, QColor, QConicalGradient, | ||||
|     QCursor, QFont, QFontDatabase, QGradient, | ||||
|     QIcon, QImage, QKeySequence, QLinearGradient, | ||||
|     QPainter, QPalette, QPixmap, QRadialGradient, | ||||
|     QTransform) | ||||
| from PySide6.QtWidgets import (QAbstractButton, QApplication, QComboBox, QDialog, | ||||
|     QDialogButtonBox, QFormLayout, QLabel, QLineEdit, | ||||
|     QSizePolicy, QStackedWidget, QVBoxLayout, QWidget) | ||||
|  | ||||
| class Ui_setupDialog(object): | ||||
|     def setupUi(self, setupDialog): | ||||
|         if not setupDialog.objectName(): | ||||
|             setupDialog.setObjectName(u"setupDialog") | ||||
|         setupDialog.resize(400, 300) | ||||
|         self.actionBrowse = QAction(setupDialog) | ||||
|         self.actionBrowse.setObjectName(u"actionBrowse") | ||||
|         icon = QIcon(QIcon.fromTheme(u"folder-open")) | ||||
|         self.actionBrowse.setIcon(icon) | ||||
|         self.actionBrowse.setMenuRole(QAction.MenuRole.NoRole) | ||||
|         self.verticalLayout = QVBoxLayout(setupDialog) | ||||
|         self.verticalLayout.setObjectName(u"verticalLayout") | ||||
|         self.comboBox = QComboBox(setupDialog) | ||||
|         self.comboBox.addItem("") | ||||
|         self.comboBox.addItem("") | ||||
|         self.comboBox.setObjectName(u"comboBox") | ||||
|  | ||||
|         self.verticalLayout.addWidget(self.comboBox) | ||||
|  | ||||
|         self.stackedWidget = QStackedWidget(setupDialog) | ||||
|         self.stackedWidget.setObjectName(u"stackedWidget") | ||||
|         self.mariadb = QWidget() | ||||
|         self.mariadb.setObjectName(u"mariadb") | ||||
|         self.formLayout = QFormLayout(self.mariadb) | ||||
|         self.formLayout.setObjectName(u"formLayout") | ||||
|         self.label = QLabel(self.mariadb) | ||||
|         self.label.setObjectName(u"label") | ||||
|  | ||||
|         self.formLayout.setWidget(0, QFormLayout.LabelRole, self.label) | ||||
|  | ||||
|         self.hostEdit = QLineEdit(self.mariadb) | ||||
|         self.hostEdit.setObjectName(u"hostEdit") | ||||
|  | ||||
|         self.formLayout.setWidget(0, QFormLayout.FieldRole, self.hostEdit) | ||||
|  | ||||
|         self.label_2 = QLabel(self.mariadb) | ||||
|         self.label_2.setObjectName(u"label_2") | ||||
|  | ||||
|         self.formLayout.setWidget(1, QFormLayout.LabelRole, self.label_2) | ||||
|  | ||||
|         self.portEdit = QLineEdit(self.mariadb) | ||||
|         self.portEdit.setObjectName(u"portEdit") | ||||
|  | ||||
|         self.formLayout.setWidget(1, QFormLayout.FieldRole, self.portEdit) | ||||
|  | ||||
|         self.label_3 = QLabel(self.mariadb) | ||||
|         self.label_3.setObjectName(u"label_3") | ||||
|  | ||||
|         self.formLayout.setWidget(2, QFormLayout.LabelRole, self.label_3) | ||||
|  | ||||
|         self.dbEdit = QLineEdit(self.mariadb) | ||||
|         self.dbEdit.setObjectName(u"dbEdit") | ||||
|  | ||||
|         self.formLayout.setWidget(2, QFormLayout.FieldRole, self.dbEdit) | ||||
|  | ||||
|         self.label_4 = QLabel(self.mariadb) | ||||
|         self.label_4.setObjectName(u"label_4") | ||||
|  | ||||
|         self.formLayout.setWidget(3, QFormLayout.LabelRole, self.label_4) | ||||
|  | ||||
|         self.userEdit = QLineEdit(self.mariadb) | ||||
|         self.userEdit.setObjectName(u"userEdit") | ||||
|  | ||||
|         self.formLayout.setWidget(3, QFormLayout.FieldRole, self.userEdit) | ||||
|  | ||||
|         self.label_5 = QLabel(self.mariadb) | ||||
|         self.label_5.setObjectName(u"label_5") | ||||
|  | ||||
|         self.formLayout.setWidget(4, QFormLayout.LabelRole, self.label_5) | ||||
|  | ||||
|         self.pwEdit = QLineEdit(self.mariadb) | ||||
|         self.pwEdit.setObjectName(u"pwEdit") | ||||
|         self.pwEdit.setEchoMode(QLineEdit.EchoMode.Password) | ||||
|  | ||||
|         self.formLayout.setWidget(4, QFormLayout.FieldRole, self.pwEdit) | ||||
|  | ||||
|         self.label_7 = QLabel(self.mariadb) | ||||
|         self.label_7.setObjectName(u"label_7") | ||||
|  | ||||
|         self.formLayout.setWidget(5, QFormLayout.LabelRole, self.label_7) | ||||
|  | ||||
|         self.pwConfirmEdit = QLineEdit(self.mariadb) | ||||
|         self.pwConfirmEdit.setObjectName(u"pwConfirmEdit") | ||||
|         self.pwConfirmEdit.setEchoMode(QLineEdit.EchoMode.Password) | ||||
|  | ||||
|         self.formLayout.setWidget(5, QFormLayout.FieldRole, self.pwConfirmEdit) | ||||
|  | ||||
|         self.stackedWidget.addWidget(self.mariadb) | ||||
|         self.sqlite = QWidget() | ||||
|         self.sqlite.setObjectName(u"sqlite") | ||||
|         self.formLayout_2 = QFormLayout(self.sqlite) | ||||
|         self.formLayout_2.setObjectName(u"formLayout_2") | ||||
|         self.label_6 = QLabel(self.sqlite) | ||||
|         self.label_6.setObjectName(u"label_6") | ||||
|  | ||||
|         self.formLayout_2.setWidget(0, QFormLayout.LabelRole, self.label_6) | ||||
|  | ||||
|         self.sqliteEdit = QLineEdit(self.sqlite) | ||||
|         self.sqliteEdit.setObjectName(u"sqliteEdit") | ||||
|  | ||||
|         self.formLayout_2.setWidget(0, QFormLayout.FieldRole, self.sqliteEdit) | ||||
|  | ||||
|         self.stackedWidget.addWidget(self.sqlite) | ||||
|  | ||||
|         self.verticalLayout.addWidget(self.stackedWidget) | ||||
|  | ||||
|         self.buttonBox = QDialogButtonBox(setupDialog) | ||||
|         self.buttonBox.setObjectName(u"buttonBox") | ||||
|         self.buttonBox.setOrientation(Qt.Orientation.Horizontal) | ||||
|         self.buttonBox.setStandardButtons(QDialogButtonBox.StandardButton.Cancel|QDialogButtonBox.StandardButton.Ok) | ||||
|  | ||||
|         self.verticalLayout.addWidget(self.buttonBox) | ||||
|  | ||||
| #if QT_CONFIG(shortcut) | ||||
|         self.label.setBuddy(self.hostEdit) | ||||
|         self.label_2.setBuddy(self.portEdit) | ||||
|         self.label_3.setBuddy(self.dbEdit) | ||||
|         self.label_4.setBuddy(self.userEdit) | ||||
|         self.label_5.setBuddy(self.pwEdit) | ||||
|         self.label_7.setBuddy(self.pwConfirmEdit) | ||||
| #endif // QT_CONFIG(shortcut) | ||||
|  | ||||
|         self.retranslateUi(setupDialog) | ||||
|         self.buttonBox.accepted.connect(setupDialog.accept) | ||||
|         self.buttonBox.rejected.connect(setupDialog.reject) | ||||
|         self.comboBox.currentIndexChanged.connect(self.stackedWidget.setCurrentIndex) | ||||
|  | ||||
|         self.stackedWidget.setCurrentIndex(0) | ||||
|  | ||||
|  | ||||
|         QMetaObject.connectSlotsByName(setupDialog) | ||||
|     # setupUi | ||||
|  | ||||
|     def retranslateUi(self, setupDialog): | ||||
|         setupDialog.setWindowTitle(QCoreApplication.translate("setupDialog", u"setupDialog", None)) | ||||
|         self.actionBrowse.setText("") | ||||
| #if QT_CONFIG(tooltip) | ||||
|         self.actionBrowse.setToolTip(QCoreApplication.translate("setupDialog", u"Browse for database file", None)) | ||||
| #endif // QT_CONFIG(tooltip) | ||||
|         self.comboBox.setItemText(0, QCoreApplication.translate("setupDialog", u"MariaDB/MySQL", None)) | ||||
|         self.comboBox.setItemText(1, QCoreApplication.translate("setupDialog", u"SQLite", None)) | ||||
|  | ||||
|         self.label.setText(QCoreApplication.translate("setupDialog", u"Host", None)) | ||||
|         self.label_2.setText(QCoreApplication.translate("setupDialog", u"Port", None)) | ||||
|         self.portEdit.setText(QCoreApplication.translate("setupDialog", u"3306", None)) | ||||
|         self.label_3.setText(QCoreApplication.translate("setupDialog", u"Database Name", None)) | ||||
|         self.dbEdit.setText(QCoreApplication.translate("setupDialog", u"scotus", None)) | ||||
|         self.label_4.setText(QCoreApplication.translate("setupDialog", u"User Name", None)) | ||||
|         self.userEdit.setText(QCoreApplication.translate("setupDialog", u"scotus", None)) | ||||
|         self.label_5.setText(QCoreApplication.translate("setupDialog", u"Password", None)) | ||||
|         self.label_7.setText(QCoreApplication.translate("setupDialog", u"Confirm", None)) | ||||
|         self.label_6.setText(QCoreApplication.translate("setupDialog", u"Datbase File", None)) | ||||
|     # retranslateUi | ||||
|  | ||||
							
								
								
									
										231
									
								
								ui/dbSetup.ui
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										231
									
								
								ui/dbSetup.ui
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,231 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <ui version="4.0"> | ||||
|  <class>setupDialog</class> | ||||
|  <widget class="QDialog" name="setupDialog"> | ||||
|   <property name="geometry"> | ||||
|    <rect> | ||||
|     <x>0</x> | ||||
|     <y>0</y> | ||||
|     <width>400</width> | ||||
|     <height>300</height> | ||||
|    </rect> | ||||
|   </property> | ||||
|   <property name="windowTitle"> | ||||
|    <string>setupDialog</string> | ||||
|   </property> | ||||
|   <layout class="QVBoxLayout" name="verticalLayout"> | ||||
|    <item> | ||||
|     <widget class="QComboBox" name="comboBox"> | ||||
|      <item> | ||||
|       <property name="text"> | ||||
|        <string>MariaDB/MySQL</string> | ||||
|       </property> | ||||
|      </item> | ||||
|      <item> | ||||
|       <property name="text"> | ||||
|        <string>SQLite</string> | ||||
|       </property> | ||||
|      </item> | ||||
|     </widget> | ||||
|    </item> | ||||
|    <item> | ||||
|     <widget class="QStackedWidget" name="stackedWidget"> | ||||
|      <property name="currentIndex"> | ||||
|       <number>0</number> | ||||
|      </property> | ||||
|      <widget class="QWidget" name="mariadb"> | ||||
|       <layout class="QFormLayout" name="formLayout"> | ||||
|        <item row="0" column="0"> | ||||
|         <widget class="QLabel" name="label"> | ||||
|          <property name="text"> | ||||
|           <string>Host</string> | ||||
|          </property> | ||||
|          <property name="buddy"> | ||||
|           <cstring>hostEdit</cstring> | ||||
|          </property> | ||||
|         </widget> | ||||
|        </item> | ||||
|        <item row="0" column="1"> | ||||
|         <widget class="QLineEdit" name="hostEdit"/> | ||||
|        </item> | ||||
|        <item row="1" column="0"> | ||||
|         <widget class="QLabel" name="label_2"> | ||||
|          <property name="text"> | ||||
|           <string>Port</string> | ||||
|          </property> | ||||
|          <property name="buddy"> | ||||
|           <cstring>portEdit</cstring> | ||||
|          </property> | ||||
|         </widget> | ||||
|        </item> | ||||
|        <item row="1" column="1"> | ||||
|         <widget class="QLineEdit" name="portEdit"> | ||||
|          <property name="text"> | ||||
|           <string>3306</string> | ||||
|          </property> | ||||
|         </widget> | ||||
|        </item> | ||||
|        <item row="2" column="0"> | ||||
|         <widget class="QLabel" name="label_3"> | ||||
|          <property name="text"> | ||||
|           <string>Database Name</string> | ||||
|          </property> | ||||
|          <property name="buddy"> | ||||
|           <cstring>dbEdit</cstring> | ||||
|          </property> | ||||
|         </widget> | ||||
|        </item> | ||||
|        <item row="2" column="1"> | ||||
|         <widget class="QLineEdit" name="dbEdit"> | ||||
|          <property name="text"> | ||||
|           <string>scotus</string> | ||||
|          </property> | ||||
|         </widget> | ||||
|        </item> | ||||
|        <item row="3" column="0"> | ||||
|         <widget class="QLabel" name="label_4"> | ||||
|          <property name="text"> | ||||
|           <string>User Name</string> | ||||
|          </property> | ||||
|          <property name="buddy"> | ||||
|           <cstring>userEdit</cstring> | ||||
|          </property> | ||||
|         </widget> | ||||
|        </item> | ||||
|        <item row="3" column="1"> | ||||
|         <widget class="QLineEdit" name="userEdit"> | ||||
|          <property name="text"> | ||||
|           <string>scotus</string> | ||||
|          </property> | ||||
|         </widget> | ||||
|        </item> | ||||
|        <item row="4" column="0"> | ||||
|         <widget class="QLabel" name="label_5"> | ||||
|          <property name="text"> | ||||
|           <string>Password</string> | ||||
|          </property> | ||||
|          <property name="buddy"> | ||||
|           <cstring>pwEdit</cstring> | ||||
|          </property> | ||||
|         </widget> | ||||
|        </item> | ||||
|        <item row="4" column="1"> | ||||
|         <widget class="QLineEdit" name="pwEdit"> | ||||
|          <property name="echoMode"> | ||||
|           <enum>QLineEdit::EchoMode::Password</enum> | ||||
|          </property> | ||||
|         </widget> | ||||
|        </item> | ||||
|        <item row="5" column="0"> | ||||
|         <widget class="QLabel" name="label_7"> | ||||
|          <property name="text"> | ||||
|           <string>Confirm</string> | ||||
|          </property> | ||||
|          <property name="buddy"> | ||||
|           <cstring>pwConfirmEdit</cstring> | ||||
|          </property> | ||||
|         </widget> | ||||
|        </item> | ||||
|        <item row="5" column="1"> | ||||
|         <widget class="QLineEdit" name="pwConfirmEdit"> | ||||
|          <property name="echoMode"> | ||||
|           <enum>QLineEdit::EchoMode::Password</enum> | ||||
|          </property> | ||||
|         </widget> | ||||
|        </item> | ||||
|       </layout> | ||||
|      </widget> | ||||
|      <widget class="QWidget" name="sqlite"> | ||||
|       <layout class="QFormLayout" name="formLayout_2"> | ||||
|        <item row="0" column="0"> | ||||
|         <widget class="QLabel" name="label_6"> | ||||
|          <property name="text"> | ||||
|           <string>Datbase File</string> | ||||
|          </property> | ||||
|         </widget> | ||||
|        </item> | ||||
|        <item row="0" column="1"> | ||||
|         <widget class="QLineEdit" name="sqliteEdit"/> | ||||
|        </item> | ||||
|       </layout> | ||||
|      </widget> | ||||
|     </widget> | ||||
|    </item> | ||||
|    <item> | ||||
|     <widget class="QDialogButtonBox" name="buttonBox"> | ||||
|      <property name="orientation"> | ||||
|       <enum>Qt::Orientation::Horizontal</enum> | ||||
|      </property> | ||||
|      <property name="standardButtons"> | ||||
|       <set>QDialogButtonBox::StandardButton::Cancel|QDialogButtonBox::StandardButton::Ok</set> | ||||
|      </property> | ||||
|     </widget> | ||||
|    </item> | ||||
|   </layout> | ||||
|   <action name="actionBrowse"> | ||||
|    <property name="icon"> | ||||
|     <iconset theme="folder-open"/> | ||||
|    </property> | ||||
|    <property name="text"> | ||||
|     <string/> | ||||
|    </property> | ||||
|    <property name="toolTip"> | ||||
|     <string>Browse for database file</string> | ||||
|    </property> | ||||
|    <property name="menuRole"> | ||||
|     <enum>QAction::MenuRole::NoRole</enum> | ||||
|    </property> | ||||
|   </action> | ||||
|  </widget> | ||||
|  <resources/> | ||||
|  <connections> | ||||
|   <connection> | ||||
|    <sender>buttonBox</sender> | ||||
|    <signal>accepted()</signal> | ||||
|    <receiver>setupDialog</receiver> | ||||
|    <slot>accept()</slot> | ||||
|    <hints> | ||||
|     <hint type="sourcelabel"> | ||||
|      <x>227</x> | ||||
|      <y>278</y> | ||||
|     </hint> | ||||
|     <hint type="destinationlabel"> | ||||
|      <x>157</x> | ||||
|      <y>274</y> | ||||
|     </hint> | ||||
|    </hints> | ||||
|   </connection> | ||||
|   <connection> | ||||
|    <sender>buttonBox</sender> | ||||
|    <signal>rejected()</signal> | ||||
|    <receiver>setupDialog</receiver> | ||||
|    <slot>reject()</slot> | ||||
|    <hints> | ||||
|     <hint type="sourcelabel"> | ||||
|      <x>295</x> | ||||
|      <y>284</y> | ||||
|     </hint> | ||||
|     <hint type="destinationlabel"> | ||||
|      <x>286</x> | ||||
|      <y>274</y> | ||||
|     </hint> | ||||
|    </hints> | ||||
|   </connection> | ||||
|   <connection> | ||||
|    <sender>comboBox</sender> | ||||
|    <signal>currentIndexChanged(int)</signal> | ||||
|    <receiver>stackedWidget</receiver> | ||||
|    <slot>setCurrentIndex(int)</slot> | ||||
|    <hints> | ||||
|     <hint type="sourcelabel"> | ||||
|      <x>199</x> | ||||
|      <y>22</y> | ||||
|     </hint> | ||||
|     <hint type="destinationlabel"> | ||||
|      <x>199</x> | ||||
|      <y>149</y> | ||||
|     </hint> | ||||
|    </hints> | ||||
|   </connection> | ||||
|  </connections> | ||||
| </ui> | ||||
| @@ -16,11 +16,12 @@ from PySide6.QtGui import (QAction, QBrush, QColor, QConicalGradient, | ||||
|     QIcon, QImage, QKeySequence, QLinearGradient, | ||||
|     QPainter, QPalette, QPixmap, QRadialGradient, | ||||
|     QTransform) | ||||
| from PySide6.QtPdfWidgets import QPdfView | ||||
| from PySide6.QtWidgets import (QApplication, QDialog, QHeaderView, QSizePolicy, | ||||
|     QSplitter, QTabWidget, QToolBar, QTreeView, | ||||
|     QVBoxLayout, QWidget) | ||||
| import ui.resources_rc | ||||
|  | ||||
| from scotusPdfView import scotusPdfView | ||||
| import resources_rc | ||||
|  | ||||
| class Ui_pdfViewer(object): | ||||
|     def setupUi(self, pdfViewer): | ||||
| @@ -107,7 +108,7 @@ class Ui_pdfViewer(object): | ||||
|         self.pagesTab.setObjectName(u"pagesTab") | ||||
|         self.tabWidget.addTab(self.pagesTab, "") | ||||
|         self.splitter.addWidget(self.tabWidget) | ||||
|         self.pdfView = QPdfView(self.splitter) | ||||
|         self.pdfView = scotusPdfView(self.splitter) | ||||
|         self.pdfView.setObjectName(u"pdfView") | ||||
|         sizePolicy1 = QSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Expanding) | ||||
|         sizePolicy1.setHorizontalStretch(10) | ||||
|   | ||||
| @@ -99,7 +99,7 @@ | ||||
|           </attribute> | ||||
|          </widget> | ||||
|         </widget> | ||||
|         <widget class="QPdfView" name="pdfView" native="true"> | ||||
|         <widget class="scotusPdfView" name="pdfView" native="true"> | ||||
|          <property name="sizePolicy"> | ||||
|           <sizepolicy hsizetype="Expanding" vsizetype="Expanding"> | ||||
|            <horstretch>10</horstretch> | ||||
| @@ -215,9 +215,9 @@ | ||||
|  </widget> | ||||
|  <customwidgets> | ||||
|   <customwidget> | ||||
|    <class>QPdfView</class> | ||||
|    <class>scotusPdfView</class> | ||||
|    <extends>QWidget</extends> | ||||
|    <header>qpdfview.h</header> | ||||
|    <header>scotusPdfView.h</header> | ||||
|   </customwidget> | ||||
|  </customwidgets> | ||||
|  <resources> | ||||
|   | ||||
							
								
								
									
										99
									
								
								workers.py
									
									
									
									
									
								
							
							
						
						
									
										99
									
								
								workers.py
									
									
									
									
									
								
							| @@ -4,7 +4,7 @@ import re | ||||
| import dateparser | ||||
| import requests | ||||
| from bs4 import BeautifulSoup, Tag | ||||
| from PySide6.QtCore import QDateTime, QThread | ||||
| from PySide6.QtCore import QDateTime, QThread, Signal | ||||
| from PySide6.QtSql import QSqlDatabase, QSqlQuery | ||||
|  | ||||
| from lib.utils import query_error | ||||
| @@ -80,7 +80,6 @@ def update_proceedings( | ||||
|                 if not query.exec(): | ||||
|                     query_error(query) | ||||
|     assert isinstance(text, str) | ||||
|     print(f"text: {text.lower()}") | ||||
|     # | ||||
|     # If cert is denied, a petion for rehearing can be requested. | ||||
|     # The petitioner has 40 days to file for a rehearing. | ||||
| @@ -107,7 +106,6 @@ def update_db(case_id: str, db: QSqlDatabase) -> int: | ||||
|     # | ||||
|     # We assume that case_id == docket_id at this point.  If it does not, | ||||
|     # then we will build out from the request we get | ||||
|     print(f"Updating {case_id}") | ||||
|     matches = re.match(r"(\d\d)[-A](\d+)(.*)$", case_id) | ||||
|     if matches is None: | ||||
|         raise Exception(f"Not a match {case_id}") | ||||
| @@ -130,12 +128,16 @@ def update_db(case_id: str, db: QSqlDatabase) -> int: | ||||
|     r = requests.get( | ||||
|         f"https://www.supremecourt.gov/docket/docketfiles/html/public/{case_id}.html" | ||||
|     ) | ||||
|     if r.status_code == 404: | ||||
|         return -1 | ||||
|  | ||||
|     if r.status_code != 200: | ||||
|         print(r.status_code) | ||||
|         exit(1) | ||||
|     bs = BeautifulSoup(r.text, "lxml") | ||||
|     # | ||||
|     # SCOTUS does not return 404 for page not found. | ||||
|     # Feb 27: SCOTUS is returning 404s but I don't trust them. | ||||
|     # | ||||
|     title = bs.find("title") | ||||
|     assert isinstance(title, Tag) and isinstance(title.string, str) | ||||
| @@ -156,9 +158,7 @@ def update_db(case_id: str, db: QSqlDatabase) -> int: | ||||
|     assert isinstance(tmp, str) | ||||
|     matches = re.match(r"(No.)?\s*(\d+[-A]\d+).*$", tmp) | ||||
|     assert matches is not None | ||||
|     print(matches, matches.groups()) | ||||
|     docket_id = matches.group(2) | ||||
|     print(f"Found {docket_id}") | ||||
|  | ||||
|     # | ||||
|     # Title is second row, first column | ||||
| @@ -223,9 +223,17 @@ def update_db(case_id: str, db: QSqlDatabase) -> int: | ||||
|     # | ||||
|     # If there is a linked case, we need to get the ID for that case. | ||||
|     if linked is not None: | ||||
|         # | ||||
|         # If this case is on the Emergency Docket and it is linked to | ||||
|         # a case on the regular docket, then this case is no longer active | ||||
|         # | ||||
|         deactivate = False | ||||
|  | ||||
|         linked = linked.replace("Linked with ", "") | ||||
|         for did in linked.split(","): | ||||
|             did = did.strip() | ||||
|             if re.match(r"\d+-\d+$", did): | ||||
|                 deactivate = True | ||||
|             query.prepare("SELECT * FROM cases WHERE docket_id = :did") | ||||
|             query.bindValue(":did", linked) | ||||
|             if not query.exec(): | ||||
| @@ -255,6 +263,14 @@ def update_db(case_id: str, db: QSqlDatabase) -> int: | ||||
|                 query.bindValue(":rhs", linked_id) | ||||
|                 if not query.exec(): | ||||
|                     query_error(query) | ||||
|         if re.match(r"\d+-\d+$", docket_id): | ||||
|             deactivate = False | ||||
|         if deactivate: | ||||
|             query.prepare("UPDATE cases SET active=0 WHERE case_id = :cid") | ||||
|             query.bindValue(":cid", case_id) | ||||
|             if not query.exec(): | ||||
|                 query_error(query) | ||||
|  | ||||
|     # | ||||
|     # XXX - Process lower courts | ||||
|     # | ||||
| @@ -272,7 +288,6 @@ class updateThread(QThread): | ||||
|  | ||||
|     def __init__(self) -> None: | ||||
|         super(updateThread, self).__init__() | ||||
|         print("updateThread: __init__(docket_id)") | ||||
|         return | ||||
|  | ||||
|     def setDocketId(self, docket_id: str) -> None: | ||||
| @@ -280,30 +295,32 @@ class updateThread(QThread): | ||||
|         return | ||||
|  | ||||
|     def run(self) -> None: | ||||
|         print(f"updateThread: running on {self.currentThread()}") | ||||
|         db = QSqlDatabase.cloneDatabase("qt_sql_default_connection", "update") | ||||
|         if not db.open(): | ||||
|             print(db.lastError()) | ||||
|             raise Exception("db.open()") | ||||
|  | ||||
|         case_id = update_db(str(self.docket_id), db) | ||||
|         update_db(str(self.docket_id), db) | ||||
|         db.close() | ||||
|         del db | ||||
|         QSqlDatabase.removeDatabase("update") | ||||
|         print(f"updateThread: run() returns {case_id}") | ||||
|         return | ||||
|  | ||||
|  | ||||
| class loadCases(QThread): | ||||
|     caseLoaded = Signal(int) | ||||
|     year = QDateTime.currentDateTime().toString("yy") | ||||
|     number = 0 | ||||
|     edocket = 0 | ||||
|  | ||||
|     def run(self) -> None: | ||||
|         db = QSqlDatabase.cloneDatabase("qt_sql_default_connection", "load") | ||||
|         if not db.open(): | ||||
|             raise Exception("db.open()") | ||||
|         year = QDateTime.currentDateTime().toString("yy") | ||||
|  | ||||
|         query = QSqlQuery(db) | ||||
|         query.prepare("SELECT * FROM history WHERE year = :year") | ||||
|         print(f"year = {year}") | ||||
|         query.bindValue(":year", year) | ||||
|         query.bindValue(":year", self.year) | ||||
|         if not query.exec(): | ||||
|             query_error(query) | ||||
|  | ||||
| @@ -312,48 +329,62 @@ class loadCases(QThread): | ||||
|                 "INSERT INTO history (year, edocket, number) " | ||||
|                 "VALUES (:year, 0, 1)" | ||||
|             ) | ||||
|             query.bindValue(":year", year) | ||||
|             query.bindValue(":year", self.year) | ||||
|             if not query.exec(): | ||||
|                 query_error(query) | ||||
|             edocket = 0 | ||||
|             number = 1 | ||||
|             self.number = 1 | ||||
|             history_id = query.lastInsertId() | ||||
|         else: | ||||
|             history_id = query.value("history_id") | ||||
|             edocket = query.value("edocket") | ||||
|             number = query.value("number") | ||||
|             self.number = query.value("number") | ||||
|         count = 0 | ||||
|  | ||||
|         while year > "00" and count < 100: | ||||
|         while self.year > "00" and count < 100: | ||||
|             self.caseLoaded.emit(count) | ||||
|             query.prepare("SELECT * FROM cases WHERE docket_id = :did") | ||||
|             if edocket == 1: | ||||
|                 docket_id = f"{year}A{number}" | ||||
|                 docket_id = f"{self.year}A{self.number}" | ||||
|             else: | ||||
|                 docket_id = f"{year}-{number}" | ||||
|                 docket_id = f"{self.year}-{self.number}" | ||||
|             print(f"Updating: {docket_id} ", end='') | ||||
|             query.bindValue(":did", docket_id) | ||||
|             if not query.exec(): | ||||
|                 query_error(query) | ||||
|             if query.next(): | ||||
|                 if query.value("active") == 0: | ||||
|                     print("Already exists and is inactive") | ||||
|                     number += 1 | ||||
|                     self.number += 1 | ||||
|                     print("INACTIVE") | ||||
|                     continue | ||||
|             print() | ||||
|             result = update_db(docket_id, db) | ||||
|             print(f"result: {result}") | ||||
|             if result < 0: | ||||
|                 year = f"{int(year) - 1:02d}" | ||||
|                 if number > 1: | ||||
|                 if edocket == 0: | ||||
|                     edocket = 1 | ||||
|                     self.number = 1 | ||||
|                     query.prepare("UPDATE history SET number = :number, " | ||||
|                                   "edocket=:edocket " | ||||
|                                   "WHERE history_id=:hid") | ||||
|                     query.bindValue(':number', 1) | ||||
|                     query.bindValue(':edocket', 1) | ||||
|                     query.bindValue(':hid', history_id) | ||||
|                     if not query.exec(): | ||||
|                         query_error(query) | ||||
|                     continue | ||||
|                 edocket = 0 | ||||
|                 self.year = f"{int(self.year) - 1:02d}" | ||||
|                 if self.number > 1: | ||||
|                     query.prepare( | ||||
|                         "UPDATE history set number = :number WHERE history_id=:hid" | ||||
|                     ) | ||||
|                     query.bindValue(":number", number - 1) | ||||
|                     query.bindValue(":number", self.number - 1) | ||||
|                     query.bindValue(":hid", history_id) | ||||
|                     if not query.exec(): | ||||
|                         query_error(query) | ||||
|  | ||||
|                 query.prepare("SELECT * FROM history WHERE year = :year") | ||||
|                 print(f"year = {year}") | ||||
|                 query.bindValue(":year", year) | ||||
|                 query.bindValue(":year", self.year) | ||||
|                 if not query.exec(): | ||||
|                     query_error(query) | ||||
|  | ||||
| @@ -362,29 +393,31 @@ class loadCases(QThread): | ||||
|                         "INSERT INTO history (year, edocket, number) " | ||||
|                         "VALUES (:year, 0, 1)" | ||||
|                     ) | ||||
|                     query.bindValue(":year", year) | ||||
|                     query.bindValue(":year", self.year) | ||||
|                     if not query.exec(): | ||||
|                         query_error(query) | ||||
|                     edocket = 0 | ||||
|                     number = 1 | ||||
|                     self.number = 1 | ||||
|                     history_id = query.lastInsertId() | ||||
|                 else: | ||||
|                     history_id = query.value("history_id") | ||||
|                     edocket = query.value("edocket") | ||||
|                     number = query.value("number") | ||||
|                     self.number = query.value("number") | ||||
|                 continue | ||||
|  | ||||
|             number += 1 | ||||
|             self.number += 1 | ||||
|             count += 1 | ||||
|         if number > 1: | ||||
|         self.number -= 1 | ||||
|         if self.number > 1: | ||||
|             query.prepare( | ||||
|                 "UPDATE history SET number= :number WHERE year = :year" | ||||
|             ) | ||||
|             query.bindValue(":number", number) | ||||
|             query.bindValue(":year", year) | ||||
|             query.bindValue(":number", self.number) | ||||
|             query.bindValue(":year", self.year) | ||||
|             if not query.exec(): | ||||
|                 query_error(query) | ||||
|         db.close() | ||||
|         del db | ||||
|         QSqlDatabase.removeDatabase("load") | ||||
|         self.caseLoaded.emit(0) | ||||
|         return | ||||
|   | ||||
		Reference in New Issue
	
	Block a user