diff --git a/docs/source/usage_shortcuts.rst b/docs/source/usage_shortcuts.rst
index 86e60afa2..4e9313510 100644
--- a/docs/source/usage_shortcuts.rst
+++ b/docs/source/usage_shortcuts.rst
@@ -77,10 +77,10 @@ Text Search Shortcuts
":kbd:`F3`", "Find the next occurrence of the search word"
":kbd:`Ctrl+F`", "Open the search bar and search for the selected word, if any is selected"
- ":kbd:`Ctrl+G`", "Find next occurrence of search word in current document"
+ ":kbd:`Ctrl+G`", "Find the next occurrence of the search word"
":kbd:`Ctrl+H`", "Open the search tool and populate with the selected word (Mac :kbd:`Cmd+=`)"
":kbd:`Ctrl+Shift+1`", "Replace selected occurrence of the search word, and move to the next"
- ":kbd:`Ctrl+Shift+G`", "Find previous occurrence of the search word"
+ ":kbd:`Ctrl+Shift+G`", "Find the previous occurrence of the search word"
":kbd:`Shift+F3`", "Find the previous occurrence of the search word"
diff --git a/novelwriter/assets/icons/typicons_dark/icons.conf b/novelwriter/assets/icons/typicons_dark/icons.conf
index d0b695725..94b4c6337 100644
--- a/novelwriter/assets/icons/typicons_dark/icons.conf
+++ b/novelwriter/assets/icons/typicons_dark/icons.conf
@@ -46,9 +46,17 @@ cross = typ_times.svg
down = typ_chevron-down.svg
edit = typ_pencil.svg
export = typ_export.svg
+fmt_bold = nw_tb-bold.svg
+fmt_italic = nw_tb-italic.svg
+fmt_mode-md = nw_tb-markdown.svg
+fmt_mode-sc = nw_tb-shortcode.svg
+fmt_strike = nw_tb-strike.svg
+fmt_subscript = nw_tb-subscript.svg
+fmt_superscript = nw_tb-superscript.svg
+fmt_underline = nw_tb-underline.svg
forward = typ_chevron-right.svg
maximise = typ_arrow-maximise.svg
-menu = typ_th-menu.svg
+menu = typ_th-dot-menu.svg
minimise = typ_arrow-minimise.svg
noncheckable = mixed_input-none.svg
proj_chapter = mixed_document-chapter.svg
diff --git a/novelwriter/assets/icons/typicons_dark/nw_tb-bold.svg b/novelwriter/assets/icons/typicons_dark/nw_tb-bold.svg
new file mode 100644
index 000000000..2eaeb846c
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_dark/nw_tb-bold.svg
@@ -0,0 +1,4 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_dark/nw_tb-italic.svg b/novelwriter/assets/icons/typicons_dark/nw_tb-italic.svg
new file mode 100644
index 000000000..0ec5fb2a1
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_dark/nw_tb-italic.svg
@@ -0,0 +1,4 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_dark/nw_tb-markdown.svg b/novelwriter/assets/icons/typicons_dark/nw_tb-markdown.svg
new file mode 100644
index 000000000..027685965
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_dark/nw_tb-markdown.svg
@@ -0,0 +1,8 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_dark/nw_tb-shortcode.svg b/novelwriter/assets/icons/typicons_dark/nw_tb-shortcode.svg
new file mode 100644
index 000000000..4ea59ec38
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_dark/nw_tb-shortcode.svg
@@ -0,0 +1,8 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_dark/nw_tb-strike.svg b/novelwriter/assets/icons/typicons_dark/nw_tb-strike.svg
new file mode 100644
index 000000000..639dd6941
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_dark/nw_tb-strike.svg
@@ -0,0 +1,4 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_dark/nw_tb-subscript.svg b/novelwriter/assets/icons/typicons_dark/nw_tb-subscript.svg
new file mode 100644
index 000000000..e0ae3587e
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_dark/nw_tb-subscript.svg
@@ -0,0 +1,5 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_dark/nw_tb-superscript.svg b/novelwriter/assets/icons/typicons_dark/nw_tb-superscript.svg
new file mode 100644
index 000000000..1d60aec27
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_dark/nw_tb-superscript.svg
@@ -0,0 +1,5 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_dark/nw_tb-underline.svg b/novelwriter/assets/icons/typicons_dark/nw_tb-underline.svg
new file mode 100644
index 000000000..3a249f4df
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_dark/nw_tb-underline.svg
@@ -0,0 +1,5 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_dark/typ_th-dot-menu.svg b/novelwriter/assets/icons/typicons_dark/typ_th-dot-menu.svg
new file mode 100644
index 000000000..19ec3bde0
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_dark/typ_th-dot-menu.svg
@@ -0,0 +1,4 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_dark/typ_th-menu.svg b/novelwriter/assets/icons/typicons_dark/typ_th-menu.svg
deleted file mode 100644
index 4bf1c92fa..000000000
--- a/novelwriter/assets/icons/typicons_dark/typ_th-menu.svg
+++ /dev/null
@@ -1,4 +0,0 @@
-
-
diff --git a/novelwriter/assets/icons/typicons_light/icons.conf b/novelwriter/assets/icons/typicons_light/icons.conf
index 22fd94562..623145114 100644
--- a/novelwriter/assets/icons/typicons_light/icons.conf
+++ b/novelwriter/assets/icons/typicons_light/icons.conf
@@ -46,9 +46,17 @@ cross = typ_times.svg
down = typ_chevron-down.svg
edit = typ_pencil.svg
export = typ_export.svg
+fmt_bold = nw_tb-bold.svg
+fmt_italic = nw_tb-italic.svg
+fmt_mode-md = nw_tb-markdown.svg
+fmt_mode-sc = nw_tb-shortcode.svg
+fmt_strike = nw_tb-strike.svg
+fmt_subscript = nw_tb-subscript.svg
+fmt_superscript = nw_tb-superscript.svg
+fmt_underline = nw_tb-underline.svg
forward = typ_chevron-right.svg
maximise = typ_arrow-maximise.svg
-menu = typ_th-menu.svg
+menu = typ_th-dot-menu.svg
minimise = typ_arrow-minimise.svg
noncheckable = mixed_input-none.svg
proj_chapter = mixed_document-chapter.svg
diff --git a/novelwriter/assets/icons/typicons_light/nw_tb-bold.svg b/novelwriter/assets/icons/typicons_light/nw_tb-bold.svg
new file mode 100644
index 000000000..3fbfbc881
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_light/nw_tb-bold.svg
@@ -0,0 +1,4 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_light/nw_tb-italic.svg b/novelwriter/assets/icons/typicons_light/nw_tb-italic.svg
new file mode 100644
index 000000000..893d8930d
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_light/nw_tb-italic.svg
@@ -0,0 +1,4 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_light/nw_tb-markdown.svg b/novelwriter/assets/icons/typicons_light/nw_tb-markdown.svg
new file mode 100644
index 000000000..24b3809d7
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_light/nw_tb-markdown.svg
@@ -0,0 +1,8 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_light/nw_tb-shortcode.svg b/novelwriter/assets/icons/typicons_light/nw_tb-shortcode.svg
new file mode 100644
index 000000000..fc19a690f
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_light/nw_tb-shortcode.svg
@@ -0,0 +1,8 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_light/nw_tb-strike.svg b/novelwriter/assets/icons/typicons_light/nw_tb-strike.svg
new file mode 100644
index 000000000..89f4b7ee5
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_light/nw_tb-strike.svg
@@ -0,0 +1,4 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_light/nw_tb-subscript.svg b/novelwriter/assets/icons/typicons_light/nw_tb-subscript.svg
new file mode 100644
index 000000000..2384f6eac
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_light/nw_tb-subscript.svg
@@ -0,0 +1,5 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_light/nw_tb-superscript.svg b/novelwriter/assets/icons/typicons_light/nw_tb-superscript.svg
new file mode 100644
index 000000000..df0aa35c1
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_light/nw_tb-superscript.svg
@@ -0,0 +1,5 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_light/nw_tb-underline.svg b/novelwriter/assets/icons/typicons_light/nw_tb-underline.svg
new file mode 100644
index 000000000..ca122269c
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_light/nw_tb-underline.svg
@@ -0,0 +1,5 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_light/typ_th-dot-menu.svg b/novelwriter/assets/icons/typicons_light/typ_th-dot-menu.svg
new file mode 100644
index 000000000..3b5d6383c
--- /dev/null
+++ b/novelwriter/assets/icons/typicons_light/typ_th-dot-menu.svg
@@ -0,0 +1,4 @@
+
+
diff --git a/novelwriter/assets/icons/typicons_light/typ_th-menu.svg b/novelwriter/assets/icons/typicons_light/typ_th-menu.svg
deleted file mode 100644
index c23c08ca1..000000000
--- a/novelwriter/assets/icons/typicons_light/typ_th-menu.svg
+++ /dev/null
@@ -1,4 +0,0 @@
-
-
diff --git a/novelwriter/config.py b/novelwriter/config.py
index 50c8c1abb..ebd0d05f7 100644
--- a/novelwriter/config.py
+++ b/novelwriter/config.py
@@ -178,9 +178,11 @@ def __init__(self) -> None:
self.spellLanguage = "en"
# State
- self.showRefPanel = True # The reference panel for the viewer is visible
- self.viewComments = True # Comments are shown in the viewer
- self.viewSynopsis = True # Synopsis is shown in the viewer
+ self.showRefPanel = True # The reference panel for the viewer is visible
+ self.showEditToolBar = False # The document editor toolbar visibility
+ self.useShortcodes = False # Use shorcodes for basic formatting
+ self.viewComments = True # Comments are shown in the viewer
+ self.viewSynopsis = True # Synopsis is shown in the viewer
# Search Bar Switches
self.searchCase = False
@@ -598,15 +600,17 @@ def loadConfig(self) -> bool:
# State
sec = "State"
- self.showRefPanel = conf.rdBool(sec, "showrefpanel", self.showRefPanel)
- self.viewComments = conf.rdBool(sec, "viewcomments", self.viewComments)
- self.viewSynopsis = conf.rdBool(sec, "viewsynopsis", self.viewSynopsis)
- self.searchCase = conf.rdBool(sec, "searchcase", self.searchCase)
- self.searchWord = conf.rdBool(sec, "searchword", self.searchWord)
- self.searchRegEx = conf.rdBool(sec, "searchregex", self.searchRegEx)
- self.searchLoop = conf.rdBool(sec, "searchloop", self.searchLoop)
- self.searchNextFile = conf.rdBool(sec, "searchnextfile", self.searchNextFile)
- self.searchMatchCap = conf.rdBool(sec, "searchmatchcap", self.searchMatchCap)
+ self.showRefPanel = conf.rdBool(sec, "showrefpanel", self.showRefPanel)
+ self.showEditToolBar = conf.rdBool(sec, "showedittoolbar", self.showEditToolBar)
+ self.useShortcodes = conf.rdBool(sec, "useshortcodes", self.useShortcodes)
+ self.viewComments = conf.rdBool(sec, "viewcomments", self.viewComments)
+ self.viewSynopsis = conf.rdBool(sec, "viewsynopsis", self.viewSynopsis)
+ self.searchCase = conf.rdBool(sec, "searchcase", self.searchCase)
+ self.searchWord = conf.rdBool(sec, "searchword", self.searchWord)
+ self.searchRegEx = conf.rdBool(sec, "searchregex", self.searchRegEx)
+ self.searchLoop = conf.rdBool(sec, "searchloop", self.searchLoop)
+ self.searchNextFile = conf.rdBool(sec, "searchnextfile", self.searchNextFile)
+ self.searchMatchCap = conf.rdBool(sec, "searchmatchcap", self.searchMatchCap)
# Deprecated Settings or Locations as of 2.0
# ToDo: These will be loaded for a few minor releases until the users have converted them
@@ -721,6 +725,8 @@ def saveConfig(self) -> bool:
conf["State"] = {
"showrefpanel": str(self.showRefPanel),
+ "showedittoolbar": str(self.showEditToolBar),
+ "useshortcodes": str(self.useShortcodes),
"viewcomments": str(self.viewComments),
"viewsynopsis": str(self.viewSynopsis),
"searchcase": str(self.searchCase),
diff --git a/novelwriter/core/options.py b/novelwriter/core/options.py
index 0dd62f066..48a0c594b 100644
--- a/novelwriter/core/options.py
+++ b/novelwriter/core/options.py
@@ -28,7 +28,7 @@
import logging
from enum import Enum
-from typing import TYPE_CHECKING, Any
+from typing import TYPE_CHECKING, Any, TypeVar
from pathlib import Path
from novelwriter.error import logException
@@ -40,6 +40,8 @@
logger = logging.getLogger(__name__)
+NWEnum = TypeVar("NWEnum", bound=Enum)
+
VALID_MAP = {
"GuiWritingStats": {
"winWidth", "winHeight", "widthCol0", "widthCol1", "widthCol2",
@@ -201,11 +203,11 @@ def getBool(self, group: str, name: str, default: bool) -> bool:
return checkBool(self._state[group].get(name, default), default)
return default
- def getEnum(self, group: str, name: str, lookup: type, default: Enum) -> Enum:
+ def getEnum(self, group: str, name: str, lookup: type, default: NWEnum) -> NWEnum:
"""Return the value mapped to an enum. Otherwise return the
default value.
"""
- if issubclass(lookup, Enum):
+ if issubclass(lookup, type(default)):
if group in self._state:
if name in self._state[group]:
value = self._state[group][name]
diff --git a/novelwriter/gui/doceditor.py b/novelwriter/gui/doceditor.py
index e9094c8b4..a3bc291c9 100644
--- a/novelwriter/gui/doceditor.py
+++ b/novelwriter/gui/doceditor.py
@@ -46,7 +46,7 @@
)
from PyQt5.QtWidgets import (
QAction, QFrame, QGridLayout, QHBoxLayout, QLabel, QLineEdit, QMenu,
- QPlainTextEdit, QPushButton, QShortcut, QToolBar, QToolButton, QWidget,
+ QPlainTextEdit, QPushButton, QShortcut, QToolBar, QToolButton, QVBoxLayout, QWidget,
qApp
)
@@ -83,6 +83,8 @@ class GuiDocEditor(QPlainTextEdit):
novelStructureChanged = pyqtSignal()
novelItemMetaChanged = pyqtSignal(str)
spellCheckStateChanged = pyqtSignal(bool)
+ closeDocumentRequest = pyqtSignal()
+ toggleFocusModeRequest = pyqtSignal()
def __init__(self, mainGui: GuiMain) -> None:
super().__init__(parent=mainGui)
@@ -135,6 +137,12 @@ def __init__(self, mainGui: GuiMain) -> None:
self.docHeader = GuiDocEditHeader(self)
self.docFooter = GuiDocEditFooter(self)
self.docSearch = GuiDocEditSearch(self)
+ self.docToolBar = GuiDocToolBar(self)
+
+ # Connect Signals
+ self.docHeader.closeDocumentRequest.connect(self._closeCurrentDocument)
+ self.docHeader.toggleToolBarRequest.connect(self._toggleToolBarVisibility)
+ self.docToolBar.requestDocAction.connect(self.docAction)
# Context Menu
self.setContextMenuPolicy(Qt.CustomContextMenu)
@@ -248,6 +256,7 @@ def updateTheme(self) -> None:
self.docSearch.updateTheme()
self.docHeader.updateTheme()
self.docFooter.updateTheme()
+ self.docToolBar.updateTheme()
return
def updateSyntaxColours(self) -> None:
@@ -400,6 +409,7 @@ def loadText(self, tHandle, tLine=None) -> bool:
qApp.processEvents()
self.setDocumentChanged(False)
self._qDocument.clearUndoRedoStacks()
+ self.docToolBar.setVisible(CONFIG.showEditToolBar)
qApp.restoreOverrideCursor()
@@ -514,6 +524,7 @@ def updateDocMargins(self) -> None:
fY = wH - fH - tB - sH
self.docHeader.setGeometry(tB, tB, tW, tH)
self.docFooter.setGeometry(tB, fY, tW, fH)
+ self.docToolBar.move(0, tH)
rH = 0
if self.docSearch.isVisible():
@@ -841,30 +852,11 @@ def insertNewBlock(self, text: str, defaultAfter: bool = True) -> bool:
return True
- def insertKeyWord(self, keyword: str) -> bool:
- """Insert a keyword in the text editor, at the cursor position.
- If the insert line is not blank, a new line is started.
- """
- if keyword not in nwKeyWords.VALID_KEYS:
- logger.error("Invalid keyword '%s'", keyword)
- return False
- logger.debug("Inserting keyword '%s'", keyword)
- state = self.insertNewBlock("%s: " % keyword)
- return state
-
def closeSearch(self) -> bool:
"""Close the search box."""
self.docSearch.closeSearch()
return self.docSearch.isVisible()
- def toggleSearch(self) -> None:
- """Toggle the visibility of the search box."""
- if self.docSearch.isVisible():
- self.docSearch.closeSearch()
- else:
- self.beginSearch()
- return
-
##
# Document Events and Maintenance
##
@@ -959,6 +951,27 @@ def updateDocInfo(self, tHandle: str) -> None:
self.updateDocMargins()
return
+ @pyqtSlot(str)
+ def insertKeyWord(self, keyword: str) -> bool:
+ """Insert a keyword in the text editor, at the cursor position.
+ If the insert line is not blank, a new line is started.
+ """
+ if keyword not in nwKeyWords.VALID_KEYS:
+ logger.error("Invalid keyword '%s'", keyword)
+ return False
+ logger.debug("Inserting keyword '%s'", keyword)
+ state = self.insertNewBlock("%s: " % keyword)
+ return state
+
+ @pyqtSlot()
+ def toggleSearch(self) -> None:
+ """Toggle the visibility of the search box."""
+ if self.docSearch.isVisible():
+ self.docSearch.closeSearch()
+ else:
+ self.beginSearch()
+ return
+
##
# Private Slots
##
@@ -1179,6 +1192,21 @@ def _updateSelCounts(self, cCount: int, wCount: int, pCount: int) -> None:
return
+ @pyqtSlot()
+ def _closeCurrentDocument(self) -> None:
+ """Close the document. Forwarded to the main Gui."""
+ self.closeDocumentRequest.emit()
+ self.docToolBar.setVisible(False)
+ return
+
+ @pyqtSlot()
+ def _toggleToolBarVisibility(self) -> None:
+ """Toggle the visibility of the tool bar."""
+ state = not self.docToolBar.isVisible()
+ self.docToolBar.setVisible(state)
+ CONFIG.showEditToolBar = state
+ return
+
##
# Search & Replace
##
@@ -2072,6 +2100,142 @@ class BackgroundWordCounterSignals(QObject):
# END Class BackgroundWordCounterSignals
+# =============================================================================================== #
+# The Formatting and Options Fold Out Menu
+# Only used by DocEditor, and is opened by the first button in the header
+# =============================================================================================== #
+
+class GuiDocToolBar(QWidget):
+
+ requestDocAction = pyqtSignal(nwDocAction)
+
+ def __init__(self, docEditor: GuiDocEditor) -> None:
+ super().__init__(parent=docEditor)
+
+ logger.debug("Create: GuiDocToolBar")
+
+ cM = CONFIG.pxInt(4)
+ tPx = int(0.8*SHARED.theme.fontPixelSize)
+ iconSize = QSize(tPx, tPx)
+ self.setContentsMargins(0, 0, 0, 0)
+
+ # General Buttons
+ # ===============
+
+ self.tbMode = QToolButton(self)
+ self.tbMode.setToolTip(self.tr("Toggle Markdown or Shortcodes Mode"))
+ self.tbMode.setIconSize(iconSize)
+ self.tbMode.setCheckable(True)
+ self.tbMode.setChecked(CONFIG.useShortcodes)
+ self.tbMode.toggled.connect(self._toggleFormatMode)
+
+ self.tbBold = QToolButton(self)
+ self.tbBold.setIconSize(iconSize)
+ self.tbBold.clicked.connect(self._formatBold)
+
+ self.tbItalic = QToolButton(self)
+ self.tbItalic.setIconSize(iconSize)
+ self.tbItalic.clicked.connect(self._formatItalic)
+
+ self.tbStrike = QToolButton(self)
+ self.tbStrike.setIconSize(iconSize)
+ self.tbStrike.clicked.connect(self._formatStrike)
+
+ self.tbUnderline = QToolButton(self)
+ self.tbUnderline.setIconSize(iconSize)
+ self.tbUnderline.clicked.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.SC_ULINE)
+ )
+
+ self.tbSuperscript = QToolButton(self)
+ self.tbSuperscript.setIconSize(iconSize)
+ self.tbSuperscript.clicked.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.SC_SUP)
+ )
+
+ self.tbSubscript = QToolButton(self)
+ self.tbSubscript.setIconSize(iconSize)
+ self.tbSubscript.clicked.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.SC_SUB)
+ )
+
+ # Assemble
+ # ========
+
+ self.outerBox = QVBoxLayout()
+ self.outerBox.addWidget(self.tbMode)
+ self.outerBox.addWidget(self.tbBold)
+ self.outerBox.addWidget(self.tbItalic)
+ self.outerBox.addWidget(self.tbStrike)
+ self.outerBox.addWidget(self.tbUnderline)
+ self.outerBox.addWidget(self.tbSuperscript)
+ self.outerBox.addWidget(self.tbSubscript)
+ self.outerBox.setContentsMargins(cM, cM, cM, cM)
+ self.outerBox.setSpacing(cM)
+
+ self.setLayout(self.outerBox)
+ self.updateTheme()
+
+ logger.debug("Ready: GuiDocToolBar")
+
+ return
+
+ def updateTheme(self) -> None:
+ """Initialise GUI elements that depend on specific settings."""
+ palette = QPalette()
+ palette.setColor(QPalette.Window, QColor(*SHARED.theme.colBack))
+ palette.setColor(QPalette.WindowText, QColor(*SHARED.theme.colText))
+ palette.setColor(QPalette.Text, QColor(*SHARED.theme.colText))
+ self.setPalette(palette)
+
+ tPx = int(0.8*SHARED.theme.fontPixelSize)
+ self.tbMode.setIcon(SHARED.theme.getToggleIcon("fmt_mode", (tPx, tPx)))
+ self.tbBold.setIcon(SHARED.theme.getIcon("fmt_bold"))
+ self.tbItalic.setIcon(SHARED.theme.getIcon("fmt_italic"))
+ self.tbStrike.setIcon(SHARED.theme.getIcon("fmt_strike"))
+ self.tbUnderline.setIcon(SHARED.theme.getIcon("fmt_underline"))
+ self.tbSuperscript.setIcon(SHARED.theme.getIcon("fmt_superscript"))
+ self.tbSubscript.setIcon(SHARED.theme.getIcon("fmt_subscript"))
+
+ return
+
+ ##
+ # Private Slots
+ ##
+
+ @pyqtSlot(bool)
+ def _toggleFormatMode(self, checked: bool) -> None:
+ """Toggle the formatting mode."""
+ CONFIG.useShortcodes = checked
+ return
+
+ @pyqtSlot()
+ def _formatBold(self):
+ """Call the bold format action."""
+ self.requestDocAction.emit(
+ nwDocAction.SC_BOLD if self.tbMode.isChecked() else nwDocAction.STRONG
+ )
+ return
+
+ @pyqtSlot()
+ def _formatItalic(self):
+ """Call the italic format action."""
+ self.requestDocAction.emit(
+ nwDocAction.SC_ITALIC if self.tbMode.isChecked() else nwDocAction.EMPH
+ )
+ return
+
+ @pyqtSlot()
+ def _formatStrike(self):
+ """Call the strikethrough format action."""
+ self.requestDocAction.emit(
+ nwDocAction.SC_STRIKE if self.tbMode.isChecked() else nwDocAction.STRIKE
+ )
+ return
+
+# END Class GuiDocToolBar
+
+
# =============================================================================================== #
# The Embedded Document Search/Replace Feature
# Only used by DocEditor, and is at a fixed position in the QTextEdit's viewport
@@ -2487,6 +2651,9 @@ def _alertSearchValid(self, isValid: bool) -> None:
class GuiDocEditHeader(QWidget):
+ closeDocumentRequest = pyqtSignal()
+ toggleToolBarRequest = pyqtSignal()
+
def __init__(self, docEditor: GuiDocEditor) -> None:
super().__init__(parent=docEditor)
@@ -2499,6 +2666,7 @@ def __init__(self, docEditor: GuiDocEditor) -> None:
fPx = int(0.9*SHARED.theme.fontPixelSize)
hSp = CONFIG.pxInt(6)
+ iconSize = QSize(fPx, fPx)
# Main Widget Settings
self.setAutoFillBackground(True)
@@ -2518,46 +2686,46 @@ def __init__(self, docEditor: GuiDocEditor) -> None:
self.itemTitle.setFont(lblFont)
# Buttons
- self.editButton = QToolButton(self)
- self.editButton.setContentsMargins(0, 0, 0, 0)
- self.editButton.setIconSize(QSize(fPx, fPx))
- self.editButton.setFixedSize(fPx, fPx)
- self.editButton.setToolButtonStyle(Qt.ToolButtonIconOnly)
- self.editButton.setVisible(False)
- self.editButton.setToolTip(self.tr("Edit document label"))
- self.editButton.clicked.connect(self._editDocument)
+ self.tbButton = QToolButton(self)
+ self.tbButton.setContentsMargins(0, 0, 0, 0)
+ self.tbButton.setIconSize(iconSize)
+ self.tbButton.setFixedSize(fPx, fPx)
+ self.tbButton.setToolButtonStyle(Qt.ToolButtonIconOnly)
+ self.tbButton.setVisible(False)
+ self.tbButton.setToolTip(self.tr("Toggle Tool Bar"))
+ self.tbButton.clicked.connect(lambda: self.toggleToolBarRequest.emit())
self.searchButton = QToolButton(self)
self.searchButton.setContentsMargins(0, 0, 0, 0)
- self.searchButton.setIconSize(QSize(fPx, fPx))
+ self.searchButton.setIconSize(iconSize)
self.searchButton.setFixedSize(fPx, fPx)
self.searchButton.setToolButtonStyle(Qt.ToolButtonIconOnly)
self.searchButton.setVisible(False)
- self.searchButton.setToolTip(self.tr("Search document"))
- self.searchButton.clicked.connect(self._searchDocument)
+ self.searchButton.setToolTip(self.tr("Search"))
+ self.searchButton.clicked.connect(self.docEditor.toggleSearch)
self.minmaxButton = QToolButton(self)
self.minmaxButton.setContentsMargins(0, 0, 0, 0)
- self.minmaxButton.setIconSize(QSize(fPx, fPx))
+ self.minmaxButton.setIconSize(iconSize)
self.minmaxButton.setFixedSize(fPx, fPx)
self.minmaxButton.setToolButtonStyle(Qt.ToolButtonIconOnly)
self.minmaxButton.setVisible(False)
self.minmaxButton.setToolTip(self.tr("Toggle Focus Mode"))
- self.minmaxButton.clicked.connect(self._minmaxDocument)
+ self.minmaxButton.clicked.connect(lambda: self.docEditor.toggleFocusModeRequest.emit())
self.closeButton = QToolButton(self)
self.closeButton.setContentsMargins(0, 0, 0, 0)
- self.closeButton.setIconSize(QSize(fPx, fPx))
+ self.closeButton.setIconSize(iconSize)
self.closeButton.setFixedSize(fPx, fPx)
self.closeButton.setToolButtonStyle(Qt.ToolButtonIconOnly)
self.closeButton.setVisible(False)
- self.closeButton.setToolTip(self.tr("Close the document"))
+ self.closeButton.setToolTip(self.tr("Close"))
self.closeButton.clicked.connect(self._closeDocument)
# Assemble Layout
self.outerBox = QHBoxLayout()
self.outerBox.setSpacing(hSp)
- self.outerBox.addWidget(self.editButton, 0)
+ self.outerBox.addWidget(self.tbButton, 0)
self.outerBox.addWidget(self.searchButton, 0)
self.outerBox.addWidget(self.itemTitle, 1)
self.outerBox.addWidget(self.minmaxButton, 0)
@@ -2583,7 +2751,7 @@ def __init__(self, docEditor: GuiDocEditor) -> None:
def updateTheme(self) -> None:
"""Update theme elements."""
- self.editButton.setIcon(SHARED.theme.getIcon("edit"))
+ self.tbButton.setIcon(SHARED.theme.getIcon("menu"))
self.searchButton.setIcon(SHARED.theme.getIcon("search"))
self.minmaxButton.setIcon(SHARED.theme.getIcon("maximise"))
self.closeButton.setIcon(SHARED.theme.getIcon("close"))
@@ -2593,7 +2761,7 @@ def updateTheme(self) -> None:
"QToolButton:hover {{border: none; background: rgba({0},{1},{2},0.2);}}"
).format(*SHARED.theme.colText)
- self.editButton.setStyleSheet(buttonStyle)
+ self.tbButton.setStyleSheet(buttonStyle)
self.searchButton.setStyleSheet(buttonStyle)
self.minmaxButton.setStyleSheet(buttonStyle)
self.closeButton.setStyleSheet(buttonStyle)
@@ -2623,7 +2791,7 @@ def setTitleFromHandle(self, tHandle: str | None) -> bool:
self._docHandle = tHandle
if tHandle is None:
self.itemTitle.setText("")
- self.editButton.setVisible(False)
+ self.tbButton.setVisible(False)
self.searchButton.setVisible(False)
self.closeButton.setVisible(False)
self.minmaxButton.setVisible(False)
@@ -2645,7 +2813,7 @@ def setTitleFromHandle(self, tHandle: str | None) -> bool:
return False
self.itemTitle.setText(nwItem.itemName)
- self.editButton.setVisible(True)
+ self.tbButton.setVisible(True)
self.searchButton.setVisible(True)
self.closeButton.setVisible(True)
self.minmaxButton.setVisible(True)
@@ -2667,34 +2835,16 @@ def updateFocusMode(self) -> None:
# Private Slots
##
- @pyqtSlot()
- def _editDocument(self) -> None:
- """Open the edit item dialog from the main GUI."""
- self.mainGui.editItemLabel(self._docHandle)
- return
-
- @pyqtSlot()
- def _searchDocument(self) -> None:
- """Toggle the visibility of the search box."""
- self.docEditor.toggleSearch()
- return
-
@pyqtSlot()
def _closeDocument(self) -> None:
"""Trigger the close editor on the main window."""
- self.mainGui.closeDocEditor()
- self.editButton.setVisible(False)
+ self.closeDocumentRequest.emit()
+ self.tbButton.setVisible(False)
self.searchButton.setVisible(False)
self.closeButton.setVisible(False)
self.minmaxButton.setVisible(False)
return
- @pyqtSlot()
- def _minmaxDocument(self) -> None:
- """Switch on or off Focus Mode."""
- self.mainGui.toggleFocusMode()
- return
-
##
# Events
##
diff --git a/novelwriter/gui/docviewer.py b/novelwriter/gui/docviewer.py
index 18dfdbc51..ff548fb68 100644
--- a/novelwriter/gui/docviewer.py
+++ b/novelwriter/gui/docviewer.py
@@ -34,8 +34,8 @@
from PyQt5.QtCore import pyqtSignal, pyqtSlot, QPoint, QSize, Qt, QUrl
from PyQt5.QtGui import (
- QColor, QCursor, QFont, QIcon, QMouseEvent, QPalette, QResizeEvent,
- QTextCursor, QTextOption
+ QColor, QCursor, QFont, QMouseEvent, QPalette, QResizeEvent, QTextCursor,
+ QTextOption
)
from PyQt5.QtWidgets import (
QAction, qApp, QFrame, QHBoxLayout, QLabel, QMenu, QScrollArea,
@@ -705,7 +705,7 @@ def __init__(self, docViewer):
self.backButton.setFixedSize(fPx, fPx)
self.backButton.setToolButtonStyle(Qt.ToolButtonIconOnly)
self.backButton.setVisible(False)
- self.backButton.setToolTip(self.tr("Go backward"))
+ self.backButton.setToolTip(self.tr("Go Backward"))
self.backButton.clicked.connect(self.docViewer.navBackward)
self.forwardButton = QToolButton(self)
@@ -714,7 +714,7 @@ def __init__(self, docViewer):
self.forwardButton.setFixedSize(fPx, fPx)
self.forwardButton.setToolButtonStyle(Qt.ToolButtonIconOnly)
self.forwardButton.setVisible(False)
- self.forwardButton.setToolTip(self.tr("Go forward"))
+ self.forwardButton.setToolTip(self.tr("Go Forward"))
self.forwardButton.clicked.connect(self.docViewer.navForward)
self.refreshButton = QToolButton(self)
@@ -723,7 +723,7 @@ def __init__(self, docViewer):
self.refreshButton.setFixedSize(fPx, fPx)
self.refreshButton.setToolButtonStyle(Qt.ToolButtonIconOnly)
self.refreshButton.setVisible(False)
- self.refreshButton.setToolTip(self.tr("Reload the document"))
+ self.refreshButton.setToolTip(self.tr("Reload"))
self.refreshButton.clicked.connect(self._refreshDocument)
self.closeButton = QToolButton(self)
@@ -732,7 +732,7 @@ def __init__(self, docViewer):
self.closeButton.setFixedSize(fPx, fPx)
self.closeButton.setToolButtonStyle(Qt.ToolButtonIconOnly)
self.closeButton.setVisible(False)
- self.closeButton.setToolTip(self.tr("Close the document"))
+ self.closeButton.setToolTip(self.tr("Close"))
self.closeButton.clicked.connect(self._closeDocument)
# Assemble Layout
@@ -1020,24 +1020,12 @@ def __init__(self, docViewer):
# Methods
##
- def updateTheme(self):
- """Update theme elements.
- """
+ def updateTheme(self) -> None:
+ """Update theme elements."""
# Icons
-
fPx = int(0.9*SHARED.theme.fontPixelSize)
-
- stickyOn = SHARED.theme.getPixmap("sticky-on", (fPx, fPx))
- stickyOff = SHARED.theme.getPixmap("sticky-off", (fPx, fPx))
- stickyIcon = QIcon()
- stickyIcon.addPixmap(stickyOn, QIcon.Normal, QIcon.On)
- stickyIcon.addPixmap(stickyOff, QIcon.Normal, QIcon.Off)
-
- bulletOn = SHARED.theme.getPixmap("bullet-on", (fPx, fPx))
- bulletOff = SHARED.theme.getPixmap("bullet-off", (fPx, fPx))
- bulletIcon = QIcon()
- bulletIcon.addPixmap(bulletOn, QIcon.Normal, QIcon.On)
- bulletIcon.addPixmap(bulletOff, QIcon.Normal, QIcon.Off)
+ stickyIcon = SHARED.theme.getToggleIcon("sticky", (fPx, fPx))
+ bulletIcon = SHARED.theme.getToggleIcon("bullet", (fPx, fPx))
self.showHide.setIcon(SHARED.theme.getIcon("reference"))
self.stickyRefs.setIcon(stickyIcon)
diff --git a/novelwriter/gui/mainmenu.py b/novelwriter/gui/mainmenu.py
index fa8b64add..e6453a823 100644
--- a/novelwriter/gui/mainmenu.py
+++ b/novelwriter/gui/mainmenu.py
@@ -30,7 +30,7 @@
from urllib.parse import urljoin
from urllib.request import pathname2url
-from PyQt5.QtCore import QUrl, pyqtSlot
+from PyQt5.QtCore import QUrl, pyqtSignal, pyqtSlot
from PyQt5.QtGui import QDesktopServices
from PyQt5.QtWidgets import QMenuBar, QAction
@@ -50,6 +50,11 @@ class GuiMainMenu(QMenuBar):
add them from this class.
"""
+ requestDocAction = pyqtSignal(nwDocAction)
+ requestDocInsert = pyqtSignal(nwDocInsert)
+ requestDocInsertText = pyqtSignal(str)
+ requestDocKeyWordInsert = pyqtSignal(str)
+
def __init__(self, mainGui: GuiMain) -> None:
super().__init__(parent=mainGui)
@@ -68,11 +73,6 @@ def __init__(self, mainGui: GuiMain) -> None:
self._buildToolsMenu()
self._buildHelpMenu()
- # Function Pointers
- self._docAction = self.mainGui.passDocumentAction
- self._docInsert = self.mainGui.docEditor.insertText
- self._insertKeyWord = self.mainGui.docEditor.insertKeyWord
-
logger.debug("Ready: GuiMainMenu")
return
@@ -209,7 +209,7 @@ def _buildDocumentMenu(self) -> None:
# Document > Close
self.aCloseDoc = self.docuMenu.addAction(self.tr("Close Document"))
self.aCloseDoc.setShortcut("Ctrl+W")
- self.aCloseDoc.triggered.connect(lambda: self.mainGui.closeDocEditor())
+ self.aCloseDoc.triggered.connect(self.mainGui.closeDocEditor)
# Document > Separator
self.docuMenu.addSeparator()
@@ -246,12 +246,12 @@ def _buildEditMenu(self) -> None:
# Edit > Undo
self.aEditUndo = self.editMenu.addAction(self.tr("Undo"))
self.aEditUndo.setShortcut("Ctrl+Z")
- self.aEditUndo.triggered.connect(lambda: self._docAction(nwDocAction.UNDO))
+ self.aEditUndo.triggered.connect(lambda: self.requestDocAction.emit(nwDocAction.UNDO))
# Edit > Redo
self.aEditRedo = self.editMenu.addAction(self.tr("Redo"))
self.aEditRedo.setShortcut("Ctrl+Y")
- self.aEditRedo.triggered.connect(lambda: self._docAction(nwDocAction.REDO))
+ self.aEditRedo.triggered.connect(lambda: self.requestDocAction.emit(nwDocAction.REDO))
# Edit > Separator
self.editMenu.addSeparator()
@@ -259,17 +259,17 @@ def _buildEditMenu(self) -> None:
# Edit > Cut
self.aEditCut = self.editMenu.addAction(self.tr("Cut"))
self.aEditCut.setShortcut("Ctrl+X")
- self.aEditCut.triggered.connect(lambda: self._docAction(nwDocAction.CUT))
+ self.aEditCut.triggered.connect(lambda: self.requestDocAction.emit(nwDocAction.CUT))
# Edit > Copy
self.aEditCopy = self.editMenu.addAction(self.tr("Copy"))
self.aEditCopy.setShortcut("Ctrl+C")
- self.aEditCopy.triggered.connect(lambda: self._docAction(nwDocAction.COPY))
+ self.aEditCopy.triggered.connect(lambda: self.requestDocAction.emit(nwDocAction.COPY))
# Edit > Paste
self.aEditPaste = self.editMenu.addAction(self.tr("Paste"))
self.aEditPaste.setShortcut("Ctrl+V")
- self.aEditPaste.triggered.connect(lambda: self._docAction(nwDocAction.PASTE))
+ self.aEditPaste.triggered.connect(lambda: self.requestDocAction.emit(nwDocAction.PASTE))
# Edit > Separator
self.editMenu.addSeparator()
@@ -277,12 +277,12 @@ def _buildEditMenu(self) -> None:
# Edit > Select All
self.aSelectAll = self.editMenu.addAction(self.tr("Select All"))
self.aSelectAll.setShortcut("Ctrl+A")
- self.aSelectAll.triggered.connect(lambda: self._docAction(nwDocAction.SEL_ALL))
+ self.aSelectAll.triggered.connect(lambda: self.requestDocAction.emit(nwDocAction.SEL_ALL))
# Edit > Select Paragraph
self.aSelectPar = self.editMenu.addAction(self.tr("Select Paragraph"))
self.aSelectPar.setShortcut("Ctrl+Shift+A")
- self.aSelectPar.triggered.connect(lambda: self._docAction(nwDocAction.SEL_PARA))
+ self.aSelectPar.triggered.connect(lambda: self.requestDocAction.emit(nwDocAction.SEL_PARA))
return
@@ -330,7 +330,7 @@ def _buildViewMenu(self) -> None:
# View > Focus Mode
self.aFocusMode = self.viewMenu.addAction(self.tr("Focus Mode"))
self.aFocusMode.setShortcut("F8")
- self.aFocusMode.triggered.connect(lambda: self.mainGui.toggleFocusMode())
+ self.aFocusMode.triggered.connect(self.mainGui.toggleFocusMode)
# View > Toggle Full Screen
self.aFullScreen = self.viewMenu.addAction(self.tr("Full Screen Mode"))
@@ -350,22 +350,30 @@ def _buildInsertMenu(self) -> None:
# Insert > Short Dash
self.aInsENDash = self.mInsDashes.addAction(self.tr("Short Dash"))
self.aInsENDash.setShortcut("Ctrl+K, -")
- self.aInsENDash.triggered.connect(lambda: self._docInsert(nwUnicode.U_ENDASH))
+ self.aInsENDash.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_ENDASH)
+ )
# Insert > Long Dash
self.aInsEMDash = self.mInsDashes.addAction(self.tr("Long Dash"))
self.aInsEMDash.setShortcut("Ctrl+K, _")
- self.aInsEMDash.triggered.connect(lambda: self._docInsert(nwUnicode.U_EMDASH))
+ self.aInsEMDash.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_EMDASH)
+ )
# Insert > Long Dash
self.aInsHorBar = self.mInsDashes.addAction(self.tr("Horizontal Bar"))
self.aInsHorBar.setShortcut("Ctrl+K, Ctrl+_")
- self.aInsHorBar.triggered.connect(lambda: self._docInsert(nwUnicode.U_HBAR))
+ self.aInsHorBar.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_HBAR)
+ )
# Insert > Figure Dash
self.aInsFigDash = self.mInsDashes.addAction(self.tr("Figure Dash"))
self.aInsFigDash.setShortcut("Ctrl+K, ~")
- self.aInsFigDash.triggered.connect(lambda: self._docInsert(nwUnicode.U_FGDASH))
+ self.aInsFigDash.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_FGDASH)
+ )
# Insert > Quote Marks
self.mInsQuotes = self.insMenu.addMenu(self.tr("Quote Marks"))
@@ -373,27 +381,37 @@ def _buildInsertMenu(self) -> None:
# Insert > Left Single Quote
self.aInsQuoteLS = self.mInsQuotes.addAction(self.tr("Left Single Quote"))
self.aInsQuoteLS.setShortcut("Ctrl+K, 1")
- self.aInsQuoteLS.triggered.connect(lambda: self._docInsert(nwDocInsert.QUOTE_LS))
+ self.aInsQuoteLS.triggered.connect(
+ lambda: self.requestDocInsert.emit(nwDocInsert.QUOTE_LS)
+ )
# Insert > Right Single Quote
self.aInsQuoteRS = self.mInsQuotes.addAction(self.tr("Right Single Quote"))
self.aInsQuoteRS.setShortcut("Ctrl+K, 2")
- self.aInsQuoteRS.triggered.connect(lambda: self._docInsert(nwDocInsert.QUOTE_RS))
+ self.aInsQuoteRS.triggered.connect(
+ lambda: self.requestDocInsert.emit(nwDocInsert.QUOTE_RS)
+ )
# Insert > Left Double Quote
self.aInsQuoteLD = self.mInsQuotes.addAction(self.tr("Left Double Quote"))
self.aInsQuoteLD.setShortcut("Ctrl+K, 3")
- self.aInsQuoteLD.triggered.connect(lambda: self._docInsert(nwDocInsert.QUOTE_LD))
+ self.aInsQuoteLD.triggered.connect(
+ lambda: self.requestDocInsert.emit(nwDocInsert.QUOTE_LD)
+ )
# Insert > Right Double Quote
self.aInsQuoteRD = self.mInsQuotes.addAction(self.tr("Right Double Quote"))
self.aInsQuoteRD.setShortcut("Ctrl+K, 4")
- self.aInsQuoteRD.triggered.connect(lambda: self._docInsert(nwDocInsert.QUOTE_RD))
+ self.aInsQuoteRD.triggered.connect(
+ lambda: self.requestDocInsert.emit(nwDocInsert.QUOTE_RD)
+ )
# Insert > Alternative Apostrophe
self.aInsMSApos = self.mInsQuotes.addAction(self.tr("Alternative Apostrophe"))
self.aInsMSApos.setShortcut("Ctrl+K, '")
- self.aInsMSApos.triggered.connect(lambda: self._docInsert(nwUnicode.U_MAPOSS))
+ self.aInsMSApos.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_MAPOSS)
+ )
# Insert > Symbols
self.mInsPunct = self.insMenu.addMenu(self.tr("General Punctuation"))
@@ -401,17 +419,23 @@ def _buildInsertMenu(self) -> None:
# Insert > Ellipsis
self.aInsEllipsis = self.mInsPunct.addAction(self.tr("Ellipsis"))
self.aInsEllipsis.setShortcut("Ctrl+K, .")
- self.aInsEllipsis.triggered.connect(lambda: self._docInsert(nwUnicode.U_HELLIP))
+ self.aInsEllipsis.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_HELLIP)
+ )
# Insert > Prime
self.aInsPrime = self.mInsPunct.addAction(self.tr("Prime"))
self.aInsPrime.setShortcut("Ctrl+K, Ctrl+'")
- self.aInsPrime.triggered.connect(lambda: self._docInsert(nwUnicode.U_PRIME))
+ self.aInsPrime.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_PRIME)
+ )
# Insert > Double Prime
self.aInsDPrime = self.mInsPunct.addAction(self.tr("Double Prime"))
self.aInsDPrime.setShortcut("Ctrl+K, Ctrl+\"")
- self.aInsDPrime.triggered.connect(lambda: self._docInsert(nwUnicode.U_DPRIME))
+ self.aInsDPrime.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_DPRIME)
+ )
# Insert > White Spaces
self.mInsSpace = self.insMenu.addMenu(self.tr("White Spaces"))
@@ -419,17 +443,23 @@ def _buildInsertMenu(self) -> None:
# Insert > Non-Breaking Space
self.aInsNBSpace = self.mInsSpace.addAction(self.tr("Non-Breaking Space"))
self.aInsNBSpace.setShortcut("Ctrl+K, Space")
- self.aInsNBSpace.triggered.connect(lambda: self._docInsert(nwUnicode.U_NBSP))
+ self.aInsNBSpace.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_NBSP)
+ )
# Insert > Thin Space
self.aInsThinSpace = self.mInsSpace.addAction(self.tr("Thin Space"))
self.aInsThinSpace.setShortcut("Ctrl+K, Shift+Space")
- self.aInsThinSpace.triggered.connect(lambda: self._docInsert(nwUnicode.U_THSP))
+ self.aInsThinSpace.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_THSP)
+ )
# Insert > Thin Non-Breaking Space
self.aInsThinNBSpace = self.mInsSpace.addAction(self.tr("Thin Non-Breaking Space"))
self.aInsThinNBSpace.setShortcut("Ctrl+K, Ctrl+Space")
- self.aInsThinNBSpace.triggered.connect(lambda: self._docInsert(nwUnicode.U_THNBSP))
+ self.aInsThinNBSpace.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_THNBSP)
+ )
# Insert > Symbols
self.mInsSymbol = self.insMenu.addMenu(self.tr("Other Symbols"))
@@ -437,42 +467,58 @@ def _buildInsertMenu(self) -> None:
# Insert > List Bullet
self.aInsBullet = self.mInsSymbol.addAction(self.tr("List Bullet"))
self.aInsBullet.setShortcut("Ctrl+K, *")
- self.aInsBullet.triggered.connect(lambda: self._docInsert(nwUnicode.U_BULL))
+ self.aInsBullet.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_BULL)
+ )
# Insert > Hyphen Bullet
self.aInsHyBull = self.mInsSymbol.addAction(self.tr("Hyphen Bullet"))
self.aInsHyBull.setShortcut("Ctrl+K, Ctrl+-")
- self.aInsHyBull.triggered.connect(lambda: self._docInsert(nwUnicode.U_HYBULL))
+ self.aInsHyBull.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_HYBULL)
+ )
# Insert > Flower Mark
self.aInsFlower = self.mInsSymbol.addAction(self.tr("Flower Mark"))
self.aInsFlower.setShortcut("Ctrl+K, Ctrl+*")
- self.aInsFlower.triggered.connect(lambda: self._docInsert(nwUnicode.U_FLOWER))
+ self.aInsFlower.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_FLOWER)
+ )
# Insert > Per Mille
self.aInsPerMille = self.mInsSymbol.addAction(self.tr("Per Mille"))
self.aInsPerMille.setShortcut("Ctrl+K, %")
- self.aInsPerMille.triggered.connect(lambda: self._docInsert(nwUnicode.U_PERMIL))
+ self.aInsPerMille.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_PERMIL)
+ )
# Insert > Degree Symbol
self.aInsDegree = self.mInsSymbol.addAction(self.tr("Degree Symbol"))
self.aInsDegree.setShortcut("Ctrl+K, Ctrl+O")
- self.aInsDegree.triggered.connect(lambda: self._docInsert(nwUnicode.U_DEGREE))
+ self.aInsDegree.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_DEGREE)
+ )
# Insert > Minus Sign
self.aInsMinus = self.mInsSymbol.addAction(self.tr("Minus Sign"))
self.aInsMinus.setShortcut("Ctrl+K, Ctrl+M")
- self.aInsMinus.triggered.connect(lambda: self._docInsert(nwUnicode.U_MINUS))
+ self.aInsMinus.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_MINUS)
+ )
# Insert > Times Sign
self.aInsTimes = self.mInsSymbol.addAction(self.tr("Times Sign"))
self.aInsTimes.setShortcut("Ctrl+K, Ctrl+X")
- self.aInsTimes.triggered.connect(lambda: self._docInsert(nwUnicode.U_TIMES))
+ self.aInsTimes.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_TIMES)
+ )
# Insert > Division
self.aInsDivide = self.mInsSymbol.addAction(self.tr("Division Sign"))
self.aInsDivide.setShortcut("Ctrl+K, Ctrl+D")
- self.aInsDivide.triggered.connect(lambda: self._docInsert(nwUnicode.U_DIVIDE))
+ self.aInsDivide.triggered.connect(
+ lambda: self.requestDocInsertText.emit(nwUnicode.U_DIVIDE)
+ )
# Insert > Tags and References
self.mInsKeywords = self.insMenu.addMenu(self.tr("Tags and References"))
@@ -491,7 +537,7 @@ def _buildInsertMenu(self) -> None:
self.mInsKWItems[keyWord][0].setText(trConst(nwLabels.KEY_NAME[keyWord]))
self.mInsKWItems[keyWord][0].setShortcut(self.mInsKWItems[keyWord][1])
self.mInsKWItems[keyWord][0].triggered.connect(
- lambda n, keyWord=keyWord: self._insertKeyWord(keyWord)
+ lambda n, keyWord=keyWord: self.requestDocKeyWordInsert.emit(keyWord)
)
self.mInsKeywords.addAction(self.mInsKWItems[keyWord][0])
@@ -501,22 +547,30 @@ def _buildInsertMenu(self) -> None:
# Insert > Synopsis Comment
self.aInsSynopsis = self.mInsComments.addAction(self.tr("Synopsis Comment"))
self.aInsSynopsis.setShortcut("Ctrl+K, S")
- self.aInsSynopsis.triggered.connect(lambda: self._docInsert(nwDocInsert.SYNOPSIS))
+ self.aInsSynopsis.triggered.connect(
+ lambda: self.requestDocInsert.emit(nwDocInsert.SYNOPSIS)
+ )
# Insert > Symbols
self.mInsBreaks = self.insMenu.addMenu(self.tr("Page Break and Space"))
# Insert > New Page
self.aInsNewPage = self.mInsBreaks.addAction(self.tr("Page Break"))
- self.aInsNewPage.triggered.connect(lambda: self._docInsert(nwDocInsert.NEW_PAGE))
+ self.aInsNewPage.triggered.connect(
+ lambda: self.requestDocInsert.emit(nwDocInsert.NEW_PAGE)
+ )
# Insert > Vertical Space (Single)
self.aInsVSpaceS = self.mInsBreaks.addAction(self.tr("Vertical Space (Single)"))
- self.aInsVSpaceS.triggered.connect(lambda: self._docInsert(nwDocInsert.VSPACE_S))
+ self.aInsVSpaceS.triggered.connect(
+ lambda: self.requestDocInsert.emit(nwDocInsert.VSPACE_S)
+ )
# Insert > Vertical Space (Multi)
self.aInsVSpaceM = self.mInsBreaks.addAction(self.tr("Vertical Space (Multi)"))
- self.aInsVSpaceM.triggered.connect(lambda: self._docInsert(nwDocInsert.VSPACE_M))
+ self.aInsVSpaceM.triggered.connect(
+ lambda: self.requestDocInsert.emit(nwDocInsert.VSPACE_M)
+ )
# Insert > Placeholder Text
self.aLipsumText = self.mInsBreaks.addAction(self.tr("Placeholder Text"))
@@ -532,17 +586,23 @@ def _buildFormatMenu(self) -> None:
# Format > Emphasis
self.aFmtEmph = self.fmtMenu.addAction(self.tr("Emphasis"))
self.aFmtEmph.setShortcut("Ctrl+I")
- self.aFmtEmph.triggered.connect(lambda: self._docAction(nwDocAction.EMPH))
+ self.aFmtEmph.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.EMPH)
+ )
# Format > Strong Emphasis
self.aFmtStrong = self.fmtMenu.addAction(self.tr("Strong Emphasis"))
self.aFmtStrong.setShortcut("Ctrl+B")
- self.aFmtStrong.triggered.connect(lambda: self._docAction(nwDocAction.STRONG))
+ self.aFmtStrong.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.STRONG)
+ )
# Format > Strikethrough
self.aFmtStrike = self.fmtMenu.addAction(self.tr("Strikethrough"))
self.aFmtStrike.setShortcut("Ctrl+D")
- self.aFmtStrike.triggered.connect(lambda: self._docAction(nwDocAction.STRIKE))
+ self.aFmtStrike.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.STRIKE)
+ )
# Edit > Separator
self.fmtMenu.addSeparator()
@@ -550,12 +610,16 @@ def _buildFormatMenu(self) -> None:
# Format > Double Quotes
self.aFmtDQuote = self.fmtMenu.addAction(self.tr("Wrap Double Quotes"))
self.aFmtDQuote.setShortcut("Ctrl+\"")
- self.aFmtDQuote.triggered.connect(lambda: self._docAction(nwDocAction.D_QUOTE))
+ self.aFmtDQuote.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.D_QUOTE)
+ )
# Format > Single Quotes
self.aFmtSQuote = self.fmtMenu.addAction(self.tr("Wrap Single Quotes"))
self.aFmtSQuote.setShortcut("Ctrl+'")
- self.aFmtSQuote.triggered.connect(lambda: self._docAction(nwDocAction.S_QUOTE))
+ self.aFmtSQuote.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.S_QUOTE)
+ )
# Format > Separator
self.fmtMenu.addSeparator()
@@ -564,28 +628,40 @@ def _buildFormatMenu(self) -> None:
self.mShortcodes = self.fmtMenu.addMenu(self.tr("More Formats ..."))
# Shortcode Italic
- self.aScItalic = self.mShortcodes.addAction(self.tr("Italics Shortcode"))
- self.aScItalic.triggered.connect(lambda: self._docAction(nwDocAction.SC_ITALIC))
+ self.aScItalic = self.mShortcodes.addAction(self.tr("Italics (Shortcode)"))
+ self.aScItalic.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.SC_ITALIC)
+ )
# Shortcode Bold
- self.aScBold = self.mShortcodes.addAction(self.tr("Bold Shortcode"))
- self.aScBold.triggered.connect(lambda: self._docAction(nwDocAction.SC_BOLD))
-
- # Shortcode Underline
- self.aScULine = self.mShortcodes.addAction(self.tr("Underline Shortcode"))
- self.aScULine.triggered.connect(lambda: self._docAction(nwDocAction.SC_ULINE))
+ self.aScBold = self.mShortcodes.addAction(self.tr("Bold (Shortcode)"))
+ self.aScBold.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.SC_BOLD)
+ )
# Shortcode Strikethrough
- self.aScStrike = self.mShortcodes.addAction(self.tr("Strikethrough Shortcode"))
- self.aScStrike.triggered.connect(lambda: self._docAction(nwDocAction.SC_STRIKE))
+ self.aScStrike = self.mShortcodes.addAction(self.tr("Strikethrough (Shortcode)"))
+ self.aScStrike.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.SC_STRIKE)
+ )
+
+ # Shortcode Underline
+ self.aScULine = self.mShortcodes.addAction(self.tr("Underline"))
+ self.aScULine.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.SC_ULINE)
+ )
# Shortcode Superscript
self.aScSuper = self.mShortcodes.addAction(self.tr("Superscript"))
- self.aScSuper.triggered.connect(lambda: self._docAction(nwDocAction.SC_SUP))
+ self.aScSuper.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.SC_SUP)
+ )
# Shortcode Subscript
self.aScSub = self.mShortcodes.addAction(self.tr("Subscript"))
- self.aScSub.triggered.connect(lambda: self._docAction(nwDocAction.SC_SUB))
+ self.aScSub.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.SC_SUB)
+ )
# Format > Separator
self.fmtMenu.addSeparator()
@@ -593,33 +669,45 @@ def _buildFormatMenu(self) -> None:
# Format > Header 1 (Partition)
self.aFmtHead1 = self.fmtMenu.addAction(self.tr("Header 1 (Partition)"))
self.aFmtHead1.setShortcut("Ctrl+1")
- self.aFmtHead1.triggered.connect(lambda: self._docAction(nwDocAction.BLOCK_H1))
+ self.aFmtHead1.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.BLOCK_H1)
+ )
# Format > Header 2 (Chapter)
self.aFmtHead2 = self.fmtMenu.addAction(self.tr("Header 2 (Chapter)"))
self.aFmtHead2.setShortcut("Ctrl+2")
- self.aFmtHead2.triggered.connect(lambda: self._docAction(nwDocAction.BLOCK_H2))
+ self.aFmtHead2.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.BLOCK_H2)
+ )
# Format > Header 3 (Scene)
self.aFmtHead3 = self.fmtMenu.addAction(self.tr("Header 3 (Scene)"))
self.aFmtHead3.setShortcut("Ctrl+3")
- self.aFmtHead3.triggered.connect(lambda: self._docAction(nwDocAction.BLOCK_H3))
+ self.aFmtHead3.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.BLOCK_H3)
+ )
# Format > Header 4 (Section)
self.aFmtHead4 = self.fmtMenu.addAction(self.tr("Header 4 (Section)"))
self.aFmtHead4.setShortcut("Ctrl+4")
- self.aFmtHead4.triggered.connect(lambda: self._docAction(nwDocAction.BLOCK_H4))
+ self.aFmtHead4.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.BLOCK_H4)
+ )
# Format > Separator
self.fmtMenu.addSeparator()
# Format > Novel Title
self.aFmtTitle = self.fmtMenu.addAction(self.tr("Novel Title"))
- self.aFmtTitle.triggered.connect(lambda: self._docAction(nwDocAction.BLOCK_TTL))
+ self.aFmtTitle.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.BLOCK_TTL)
+ )
# Format > Unnumbered Chapter
self.aFmtUnNum = self.fmtMenu.addAction(self.tr("Unnumbered Chapter"))
- self.aFmtUnNum.triggered.connect(lambda: self._docAction(nwDocAction.BLOCK_UNN))
+ self.aFmtUnNum.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.BLOCK_UNN)
+ )
# Format > Separator
self.fmtMenu.addSeparator()
@@ -627,17 +715,23 @@ def _buildFormatMenu(self) -> None:
# Format > Align Left
self.aFmtAlignLeft = self.fmtMenu.addAction(self.tr("Align Left"))
self.aFmtAlignLeft.setShortcut("Ctrl+5")
- self.aFmtAlignLeft.triggered.connect(lambda: self._docAction(nwDocAction.ALIGN_L))
+ self.aFmtAlignLeft.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.ALIGN_L)
+ )
# Format > Align Centre
self.aFmtAlignCentre = self.fmtMenu.addAction(self.tr("Align Centre"))
self.aFmtAlignCentre.setShortcut("Ctrl+6")
- self.aFmtAlignCentre.triggered.connect(lambda: self._docAction(nwDocAction.ALIGN_C))
+ self.aFmtAlignCentre.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.ALIGN_C)
+ )
# Format > Align Right
self.aFmtAlignRight = self.fmtMenu.addAction(self.tr("Align Right"))
self.aFmtAlignRight.setShortcut("Ctrl+7")
- self.aFmtAlignRight.triggered.connect(lambda: self._docAction(nwDocAction.ALIGN_R))
+ self.aFmtAlignRight.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.ALIGN_R)
+ )
# Format > Separator
self.fmtMenu.addSeparator()
@@ -645,12 +739,16 @@ def _buildFormatMenu(self) -> None:
# Format > Indent Left
self.aFmtIndentLeft = self.fmtMenu.addAction(self.tr("Indent Left"))
self.aFmtIndentLeft.setShortcut("Ctrl+8")
- self.aFmtIndentLeft.triggered.connect(lambda: self._docAction(nwDocAction.INDENT_L))
+ self.aFmtIndentLeft.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.INDENT_L)
+ )
# Format > Indent Right
self.aFmtIndentRight = self.fmtMenu.addAction(self.tr("Indent Right"))
self.aFmtIndentRight.setShortcut("Ctrl+9")
- self.aFmtIndentRight.triggered.connect(lambda: self._docAction(nwDocAction.INDENT_R))
+ self.aFmtIndentRight.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.INDENT_R)
+ )
# Format > Separator
self.fmtMenu.addSeparator()
@@ -658,27 +756,37 @@ def _buildFormatMenu(self) -> None:
# Format > Comment
self.aFmtComment = self.fmtMenu.addAction(self.tr("Toggle Comment"))
self.aFmtComment.setShortcut("Ctrl+/")
- self.aFmtComment.triggered.connect(lambda: self._docAction(nwDocAction.BLOCK_COM))
+ self.aFmtComment.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.BLOCK_COM)
+ )
# Format > Remove Block Format
self.aFmtNoFormat = self.fmtMenu.addAction(self.tr("Remove Block Format"))
self.aFmtNoFormat.setShortcuts(["Ctrl+0", "Ctrl+Shift+/"])
- self.aFmtNoFormat.triggered.connect(lambda: self._docAction(nwDocAction.BLOCK_TXT))
+ self.aFmtNoFormat.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.BLOCK_TXT)
+ )
# Format > Separator
self.fmtMenu.addSeparator()
# Format > Replace Single Quotes
self.aFmtReplSng = self.fmtMenu.addAction(self.tr("Convert Single Quotes"))
- self.aFmtReplSng.triggered.connect(lambda: self._docAction(nwDocAction.REPL_SNG))
+ self.aFmtReplSng.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.REPL_SNG)
+ )
# Format > Replace Double Quotes
self.aFmtReplDbl = self.fmtMenu.addAction(self.tr("Convert Double Quotes"))
- self.aFmtReplDbl.triggered.connect(lambda: self._docAction(nwDocAction.REPL_DBL))
+ self.aFmtReplDbl.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.REPL_DBL)
+ )
# Format > Remove In-Paragraph Breaks
self.aFmtRmBreaks = self.fmtMenu.addAction(self.tr("Remove In-Paragraph Breaks"))
- self.aFmtRmBreaks.triggered.connect(lambda: self._docAction(nwDocAction.RM_BREAKS))
+ self.aFmtRmBreaks.triggered.connect(
+ lambda: self.requestDocAction.emit(nwDocAction.RM_BREAKS)
+ )
return
diff --git a/novelwriter/gui/noveltree.py b/novelwriter/gui/noveltree.py
index 0742ab548..40aae9efc 100644
--- a/novelwriter/gui/noveltree.py
+++ b/novelwriter/gui/noveltree.py
@@ -29,9 +29,10 @@
from enum import Enum
from time import time
+from typing import TYPE_CHECKING
-from PyQt5.QtGui import QFont, QPalette
-from PyQt5.QtCore import Qt, QSize, pyqtSlot, pyqtSignal
+from PyQt5.QtGui import QFocusEvent, QFont, QMouseEvent, QPalette, QResizeEvent
+from PyQt5.QtCore import QModelIndex, QPoint, Qt, QSize, pyqtSlot, pyqtSignal
from PyQt5.QtWidgets import (
QAbstractItemView, QActionGroup, QFrame, QHBoxLayout, QHeaderView,
QInputDialog, QMenu, QSizePolicy, QToolButton, QToolTip, QTreeWidget,
@@ -42,8 +43,12 @@
from novelwriter.enum import nwDocMode, nwItemClass, nwOutline
from novelwriter.common import minmax
from novelwriter.constants import nwHeaders, nwKeyWords, nwLabels, trConst
+from novelwriter.core.index import IndexHeading
from novelwriter.extensions.novelselector import NovelSelector
+if TYPE_CHECKING: # pragma: no cover
+ from novelwriter.guimain import GuiMain
+
logger = logging.getLogger(__name__)
@@ -63,7 +68,7 @@ class GuiNovelView(QWidget):
selectedItemChanged = pyqtSignal(str)
openDocumentRequest = pyqtSignal(str, Enum, str, bool)
- def __init__(self, mainGui):
+ def __init__(self, mainGui: GuiMain) -> None:
super().__init__(parent=mainGui)
self.mainGui = mainGui
@@ -92,33 +97,29 @@ def __init__(self, mainGui):
# Methods
##
- def updateTheme(self):
- """Update theme elements.
- """
+ def updateTheme(self) -> None:
+ """Update theme elements."""
self.novelBar.updateTheme()
self.novelTree.updateTheme()
self.refreshTree()
return
- def initSettings(self):
- """Initialise GUI elements that depend on specific settings.
- """
+ def initSettings(self) -> None:
+ """Initialise GUI elements that depend on specific settings."""
self.novelTree.initSettings()
return
- def clearNovelView(self):
- """Clear project-related GUI content.
- """
+ def clearNovelView(self) -> None:
+ """Clear project-related GUI content."""
self.novelTree.clearContent()
self.novelBar.clearContent()
self.novelBar.setEnabled(False)
return
- def openProjectTasks(self):
- """Run open project tasks.
- """
+ def openProjectTasks(self) -> None:
+ """Run open project tasks."""
lastNovel = SHARED.project.data.getLastHandle("novelTree")
- if lastNovel not in SHARED.project.tree:
+ if lastNovel and lastNovel not in SHARED.project.tree:
lastNovel = SHARED.project.tree.findRoot(nwItemClass.NOVEL)
logger.debug("Setting novel tree to root item '%s'", lastNovel)
@@ -140,7 +141,7 @@ def openProjectTasks(self):
return
- def closeProjectTasks(self):
+ def closeProjectTasks(self) -> None:
"""Run closing project tasks."""
lastColType = self.novelTree.lastColType
lastColSize = self.novelTree.lastColSize
@@ -150,15 +151,13 @@ def closeProjectTasks(self):
self.clearNovelView()
return
- def setTreeFocus(self):
- """Set the focus to the tree widget.
- """
+ def setTreeFocus(self) -> None:
+ """Set the focus to the tree widget."""
self.novelTree.setFocus()
return
- def treeHasFocus(self):
- """Check if the novel tree has focus.
- """
+ def treeHasFocus(self) -> bool:
+ """Check if the novel tree has focus."""
return self.novelTree.hasFocus()
##
@@ -166,21 +165,19 @@ def treeHasFocus(self):
##
@pyqtSlot()
- def refreshTree(self):
- """Refresh the current tree.
- """
+ def refreshTree(self) -> None:
+ """Refresh the current tree."""
self.novelTree.refreshTree(rootHandle=SHARED.project.data.getLastHandle("novelTree"))
return
@pyqtSlot(str)
- def updateRootItem(self, tHandle):
- """If any root item changes, rebuild the novel root menu.
- """
+ def updateRootItem(self, tHandle: str) -> None:
+ """If any root item changes, rebuild the novel root menu."""
self.novelBar.buildNovelRootMenu()
return
@pyqtSlot(str)
- def updateNovelItemMeta(self, tHandle):
+ def updateNovelItemMeta(self, tHandle: str) -> None:
"""The meta data of a novel item has changed, and the tree item
needs to be refreshed.
"""
@@ -192,7 +189,7 @@ def updateNovelItemMeta(self, tHandle):
class GuiNovelToolBar(QWidget):
- def __init__(self, novelView):
+ def __init__(self, novelView: GuiNovelView) -> None:
super().__init__(parent=novelView)
logger.debug("Create: GuiNovelToolBar")
@@ -269,9 +266,8 @@ def __init__(self, novelView):
# Methods
##
- def updateTheme(self):
- """Update theme elements.
- """
+ def updateTheme(self) -> None:
+ """Update theme elements."""
# Icons
self.tbNovel.setIcon(SHARED.theme.getIcon("cls_novel"))
self.tbRefresh.setIcon(SHARED.theme.getIcon("refresh"))
@@ -287,10 +283,11 @@ def updateTheme(self):
"QToolButton {{padding: {0}px; border: none; background: transparent;}} "
"QToolButton:hover {{border: none; background: rgba({1},{2},{3},0.2);}}"
).format(CONFIG.pxInt(2), fadeCol.red(), fadeCol.green(), fadeCol.blue())
+ buttonStyleMenu = f"{buttonStyle} QToolButton::menu-indicator {{image: none;}}"
self.tbNovel.setStyleSheet(buttonStyle)
self.tbRefresh.setStyleSheet(buttonStyle)
- self.tbMore.setStyleSheet(buttonStyle)
+ self.tbMore.setStyleSheet(buttonStyleMenu)
self.novelValue.setStyleSheet(
"QComboBox {border-style: none; padding-left: 0;} "
@@ -301,30 +298,26 @@ def updateTheme(self):
return
- def clearContent(self):
- """Run clearing project tasks.
- """
+ def clearContent(self) -> None:
+ """Run clearing project tasks."""
self.novelValue.clear()
self.novelValue.setToolTip("")
return
- def buildNovelRootMenu(self):
- """Build the novel root menu.
- """
+ def buildNovelRootMenu(self) -> None:
+ """Build the novel root menu."""
self.novelValue.updateList(prefix=self.novelPrefix)
self.tbNovel.setVisible(self.novelValue.count() > 1)
return
- def setCurrentRoot(self, rootHandle):
- """Set the current active root handle.
- """
+ def setCurrentRoot(self, rootHandle: str | None) -> None:
+ """Set the current active root handle."""
self.novelValue.setHandle(rootHandle)
self.novelView.novelTree.refreshTree(rootHandle=rootHandle, overRide=True)
return
- def setLastColType(self, colType, doRefresh=True):
- """Set the last column type.
- """
+ def setLastColType(self, colType: NovelTreeColumn, doRefresh: bool = True) -> None:
+ """Set the last column type."""
self.aLastCol[colType].setChecked(True)
self.novelView.novelTree.setLastColType(colType, doRefresh=doRefresh)
return
@@ -334,24 +327,21 @@ def setLastColType(self, colType, doRefresh=True):
##
@pyqtSlot()
- def _openNovelSelector(self):
- """Trigger the dropdown list of the novel selector.
- """
+ def _openNovelSelector(self) -> None:
+ """Trigger the dropdown list of the novel selector."""
self.novelValue.showPopup()
return
@pyqtSlot()
- def _refreshNovelTree(self):
- """Rebuild the current tree.
- """
+ def _refreshNovelTree(self) -> None:
+ """Rebuild the current tree."""
rootHandle = SHARED.project.data.getLastHandle("novelTree")
self.novelView.novelTree.refreshTree(rootHandle=rootHandle, overRide=True)
return
@pyqtSlot()
- def _selectLastColumnSize(self):
- """Set the maximum width for the last column.
- """
+ def _selectLastColumnSize(self) -> None:
+ """Set the maximum width for the last column."""
oldSize = self.novelView.novelTree.lastColSize
newSize, isOk = QInputDialog.getInt(
self, self.tr("Column Size"), self.tr("Maximum column size in %"), oldSize, 15, 75, 5
@@ -365,9 +355,8 @@ def _selectLastColumnSize(self):
# Internal Functions
##
- def _addLastColAction(self, colType, actionLabel):
- """Add a column selection entry to the last column menu.
- """
+ def _addLastColAction(self, colType, actionLabel) -> None:
+ """Add a column selection entry to the last column menu."""
aLast = self.mLastCol.addAction(actionLabel)
aLast.setCheckable(True)
aLast.setActionGroup(self.gLastCol)
@@ -391,7 +380,7 @@ class GuiNovelTree(QTreeWidget):
D_KEY = Qt.ItemDataRole.UserRole + 2
D_EXTRA = Qt.ItemDataRole.UserRole + 3
- def __init__(self, novelView):
+ def __init__(self, novelView: GuiNovelView) -> None:
super().__init__(parent=novelView)
logger.debug("Create: GuiNovelTree")
@@ -461,9 +450,8 @@ def __init__(self, novelView):
return
- def initSettings(self):
- """Set or update tree widget settings.
- """
+ def initSettings(self) -> None:
+ """Set or update tree widget settings."""
# Scroll bars
if CONFIG.hideVScroll:
self.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
@@ -477,11 +465,10 @@ def initSettings(self):
return
- def updateTheme(self):
- """Update theme elements.
- """
+ def updateTheme(self) -> None:
+ """Update theme elements."""
iPx = SHARED.theme.baseIconSize
- self._pMore = SHARED.theme.loadDecoration("deco_doc_more", pxH=iPx)
+ self._pMore = SHARED.theme.loadDecoration("deco_doc_more", h=iPx)
return
##
@@ -489,28 +476,26 @@ def updateTheme(self):
##
@property
- def lastColType(self):
+ def lastColType(self) -> NovelTreeColumn:
return self._lastCol
@property
- def lastColSize(self):
+ def lastColSize(self) -> int:
return int(self._lastColSize * 100)
##
# Class Methods
##
- def clearContent(self):
- """Clear the GUI content and the related maps.
- """
+ def clearContent(self) -> None:
+ """Clear the GUI content and the related maps."""
self.clear()
self._treeMap = {}
self._lastBuild = 0
return
- def refreshTree(self, rootHandle=None, overRide=False):
- """Refresh the tree if it has been changed.
- """
+ def refreshTree(self, rootHandle: str | None = None, overRide: bool = False) -> None:
+ """Refresh the tree if it has been changed."""
logger.debug("Requesting refresh of the novel tree")
if rootHandle is None:
rootHandle = SHARED.project.tree.findRoot(nwItemClass.NOVEL)
@@ -534,9 +519,8 @@ def refreshTree(self, rootHandle=None, overRide=False):
return
- def refreshHandle(self, tHandle):
- """Refresh the data for a given handle.
- """
+ def refreshHandle(self, tHandle: str) -> None:
+ """Refresh the data for a given handle."""
idxData = SHARED.project.index.getItemData(tHandle)
if idxData is None:
return
@@ -554,7 +538,7 @@ def refreshHandle(self, tHandle):
return
- def getSelectedHandle(self):
+ def getSelectedHandle(self) -> tuple[str | None, str | None]:
"""Get the currently selected or active handle. If multiple
items are selected, return the first.
"""
@@ -566,9 +550,8 @@ def getSelectedHandle(self):
return tHandle, sTitle
return None, None
- def setLastColType(self, colType, doRefresh=True):
- """Change the content type of the last column and rebuild.
- """
+ def setLastColType(self, colType: NovelTreeColumn, doRefresh: bool = True) -> None:
+ """Change the content type of the last column and rebuild."""
if self._lastCol != colType:
logger.debug("Changing last column to %s", colType.name)
self._lastCol = colType
@@ -578,15 +561,13 @@ def setLastColType(self, colType, doRefresh=True):
self.refreshTree(rootHandle=lastNovel, overRide=True)
return
- def setLastColSize(self, colSize):
- """Set the column size in integer values between 15 and 75.
- """
+ def setLastColSize(self, colSize: int) -> None:
+ """Set the column size in integer values between 15 and 75."""
self._lastColSize = minmax(colSize, 15, 75)/100.0
return
- def setActiveHandle(self, tHandle):
- """Highlight the rows associated with a given handle.
- """
+ def setActiveHandle(self, tHandle: str | None) -> None:
+ """Highlight the rows associated with a given handle."""
tStart = time()
self._actHandle = tHandle
@@ -612,20 +593,20 @@ def setActiveHandle(self, tHandle):
# Events
##
- def mousePressEvent(self, theEvent):
+ def mousePressEvent(self, event: QMouseEvent) -> None:
"""Overload mousePressEvent to clear selection if clicking the
mouse in a blank area of the tree view, and to load a document
for viewing if the user middle-clicked.
"""
- super().mousePressEvent(theEvent)
+ super().mousePressEvent(event)
- if theEvent.button() == Qt.LeftButton:
- selItem = self.indexAt(theEvent.pos())
+ if event.button() == Qt.LeftButton:
+ selItem = self.indexAt(event.pos())
if not selItem.isValid():
self.clearSelection()
- elif theEvent.button() == Qt.MiddleButton:
- selItem = self.itemAt(theEvent.pos())
+ elif event.button() == Qt.MiddleButton:
+ selItem = self.itemAt(event.pos())
if not isinstance(selItem, QTreeWidgetItem):
return
@@ -637,16 +618,14 @@ def mousePressEvent(self, theEvent):
return
- def focusOutEvent(self, theEvent):
- """Clear the selection when the tree no longer has focus.
- """
- super().focusOutEvent(theEvent)
+ def focusOutEvent(self, event: QFocusEvent) -> None:
+ """Clear the selection when the tree no longer has focus."""
+ super().focusOutEvent(event)
self.clearSelection()
return
- def resizeEvent(self, event):
- """Elide labels in the extra column.
- """
+ def resizeEvent(self, event: QResizeEvent) -> None:
+ """Elide labels in the extra column."""
super().resizeEvent(event)
newW = event.size().width()
oldW = event.oldSize().width()
@@ -665,18 +644,17 @@ def resizeEvent(self, event):
##
@pyqtSlot("QModelIndex")
- def _treeItemClicked(self, mIndex):
- """The user clicked on an item in the tree.
- """
- if mIndex.column() == self.C_MORE:
- tHandle = mIndex.siblingAtColumn(self.C_DATA).data(self.D_HANDLE)
- sTitle = mIndex.siblingAtColumn(self.C_DATA).data(self.D_TITLE)
- tipPos = self.mapToGlobal(self.visualRect(mIndex).topRight())
+ def _treeItemClicked(self, index: QModelIndex) -> None:
+ """The user clicked on an item in the tree."""
+ if index.column() == self.C_MORE:
+ tHandle = index.siblingAtColumn(self.C_DATA).data(self.D_HANDLE)
+ sTitle = index.siblingAtColumn(self.C_DATA).data(self.D_TITLE)
+ tipPos = self.mapToGlobal(self.visualRect(index).topRight())
self._popMetaBox(tipPos, tHandle, sTitle)
return
@pyqtSlot()
- def _treeSelectionChange(self):
+ def _treeSelectionChange(self) -> None:
"""Extract the handle and line number of the currently selected
title, and send it to the tree meta panel.
"""
@@ -686,7 +664,7 @@ def _treeSelectionChange(self):
return
@pyqtSlot("QTreeWidgetItem*", int)
- def _treeDoubleClick(self, tItem, colNo):
+ def _treeDoubleClick(self, item: QTreeWidgetItem, column: int) -> None:
"""Extract the handle and line number of the title double-
clicked, and send it to the main gui class for opening in the
document editor.
@@ -699,9 +677,8 @@ def _treeDoubleClick(self, tItem, colNo):
# Internal Functions
##
- def _populateTree(self, rootHandle):
- """Build the tree based on the project index.
- """
+ def _populateTree(self, rootHandle: str | None) -> None:
+ """Build the tree based on the project index."""
self.clearContent()
tStart = time()
logger.debug("Building novel tree for root item '%s'", rootHandle)
@@ -728,9 +705,9 @@ def _populateTree(self, rootHandle):
return
- def _updateTreeItemValues(self, trItem, idxItem, tHandle, sTitle):
- """Set the tree item values from the index entry.
- """
+ def _updateTreeItemValues(self, trItem: QTreeWidgetItem, idxItem: IndexHeading,
+ tHandle: str, sTitle: str) -> None:
+ """Set the tree item values from the index entry."""
iLevel = nwHeaders.H_LEVEL.get(idxItem.level, 0)
hDec = SHARED.theme.getHeaderDecoration(iLevel)
@@ -750,9 +727,8 @@ def _updateTreeItemValues(self, trItem, idxItem, tHandle, sTitle):
return
- def _getLastColumnText(self, tHandle, sTitle):
- """Generate the text for the last column based on user settings.
- """
+ def _getLastColumnText(self, tHandle: str, sTitle: str) -> tuple[str, str]:
+ """Generate text for the last column based on user settings."""
if self._lastCol == NovelTreeColumn.HIDDEN:
return "", ""
@@ -777,14 +753,15 @@ def _getLastColumnText(self, tHandle, sTitle):
return "", ""
- def _popMetaBox(self, qPos, tHandle, sTitle):
- """Show the novel meta data box.
- """
+ def _popMetaBox(self, qPos: QPoint, tHandle: str, sTitle: str) -> None:
+ """Show the novel meta data box."""
logger.debug("Generating meta data tooltip for '%s:%s'", tHandle, sTitle)
pIndex = SHARED.project.index
novIdx = pIndex.getItemHeader(tHandle, sTitle)
refTags = pIndex.getReferences(tHandle, sTitle)
+ if not novIdx:
+ return
synopText = novIdx.synopsis
if synopText:
@@ -814,9 +791,8 @@ def _popMetaBox(self, qPos, tHandle, sTitle):
return
@staticmethod
- def _appendMetaTag(refs, key, lines):
- """Generate a reference list for a given reference key.
- """
+ def _appendMetaTag(refs: dict, key: str, lines: list[str]) -> list[str]:
+ """Generate a reference list for a given reference key."""
tags = ", ".join(refs.get(key, []))
if tags:
lines.append(f"{trConst(nwLabels.KEY_NAME[key])}: {tags}")
diff --git a/novelwriter/gui/outline.py b/novelwriter/gui/outline.py
index 84234ccfe..596a422b0 100644
--- a/novelwriter/gui/outline.py
+++ b/novelwriter/gui/outline.py
@@ -261,6 +261,7 @@ def updateTheme(self) -> None:
self.novelValue.updateList(includeAll=True)
self.aRefresh.setIcon(SHARED.theme.getIcon("refresh"))
self.tbColumns.setIcon(SHARED.theme.getIcon("menu"))
+ self.tbColumns.setStyleSheet("QToolButton::menu-indicator {image: none;}")
return
def populateNovelList(self) -> None:
diff --git a/novelwriter/gui/projtree.py b/novelwriter/gui/projtree.py
index 787f5671d..d1c2f903d 100644
--- a/novelwriter/gui/projtree.py
+++ b/novelwriter/gui/projtree.py
@@ -361,12 +361,13 @@ def updateTheme(self) -> None:
"QToolButton {{padding: {0}px; border: none; background: transparent;}} "
"QToolButton:hover {{border: none; background: rgba({1},{2},{3},0.2);}}"
).format(CONFIG.pxInt(2), fadeCol.red(), fadeCol.green(), fadeCol.blue())
+ buttonStyleMenu = f"{buttonStyle} QToolButton::menu-indicator {{image: none;}}"
- self.tbQuick.setStyleSheet(buttonStyle)
+ self.tbQuick.setStyleSheet(buttonStyleMenu)
self.tbMoveU.setStyleSheet(buttonStyle)
self.tbMoveD.setStyleSheet(buttonStyle)
- self.tbAdd.setStyleSheet(buttonStyle)
- self.tbMore.setStyleSheet(buttonStyle)
+ self.tbAdd.setStyleSheet(buttonStyleMenu)
+ self.tbMore.setStyleSheet(buttonStyleMenu)
self.tbQuick.setIcon(SHARED.theme.getIcon("bookmark"))
self.tbMoveU.setIcon(SHARED.theme.getIcon("up"))
diff --git a/novelwriter/gui/sidebar.py b/novelwriter/gui/sidebar.py
index 4b7b4e7bf..be719a42d 100644
--- a/novelwriter/gui/sidebar.py
+++ b/novelwriter/gui/sidebar.py
@@ -115,7 +115,6 @@ def __init__(self, mainGui: GuiMain) -> None:
self.outerBox.setSpacing(CONFIG.pxInt(4))
self.setLayout(self.outerBox)
-
self.updateTheme()
logger.debug("Ready: GuiSideBar")
diff --git a/novelwriter/gui/theme.py b/novelwriter/gui/theme.py
index da8f1be4f..6c4d7e767 100644
--- a/novelwriter/gui/theme.py
+++ b/novelwriter/gui/theme.py
@@ -36,7 +36,7 @@
)
from novelwriter import CONFIG
-from novelwriter.enum import nwItemLayout, nwItemType
+from novelwriter.enum import nwItemClass, nwItemLayout, nwItemType
from novelwriter.error import logException
from novelwriter.common import NWConfigParser, minmax
from novelwriter.constants import nwLabels
@@ -51,7 +51,7 @@
class GuiTheme:
- def __init__(self):
+ def __init__(self) -> None:
self.iconCache = GuiIcons(self)
@@ -130,6 +130,7 @@ def __init__(self):
self.getIcon = self.iconCache.getIcon
self.getPixmap = self.iconCache.getPixmap
self.getItemIcon = self.iconCache.getItemIcon
+ self.getToggleIcon = self.iconCache.getToggleIcon
self.loadDecoration = self.iconCache.loadDecoration
self.getHeaderDecoration = self.iconCache.getHeaderDecoration
@@ -182,7 +183,7 @@ def getTextWidth(self, text: str, font: QFont | None = None) -> int:
# Theme Methods
##
- def loadTheme(self):
+ def loadTheme(self) -> bool:
"""Load the currently specified GUI theme."""
guiTheme = CONFIG.guiTheme
if guiTheme not in self._availThemes:
@@ -269,7 +270,7 @@ def loadTheme(self):
return True
- def loadSyntax(self):
+ def loadSyntax(self) -> bool:
"""Load the currently specified syntax highlighter theme."""
guiSyntax = CONFIG.guiSyntax
if guiSyntax not in self._availSyntax:
@@ -363,7 +364,7 @@ def listSyntax(self) -> list[tuple[str, str]]:
# Internal Functions
##
- def _setGuiFont(self):
+ def _setGuiFont(self) -> None:
"""Update the GUI's font style from settings."""
theFont = QFont()
fontDB = QFontDatabase()
@@ -395,9 +396,7 @@ def _listConf(self, targetDict: dict, checkDir: Path) -> bool:
return True
- def _parseColour(
- self, parser: NWConfigParser, section: str, name: str
- ) -> list[int]:
+ def _parseColour(self, parser: NWConfigParser, section: str, name: str) -> list[int]:
"""Parse a colour value from a config string."""
if parser.has_option(section, name):
values = parser.get(section, name).split(",")
@@ -414,9 +413,8 @@ def _parseColour(
result = [0, 0, 0]
return result
- def _setPalette(
- self, parser: NWConfigParser, section: str, name: str, value: QPalette.ColorRole
- ):
+ def _setPalette(self, parser: NWConfigParser, section: str,
+ name: str, value: QPalette.ColorRole) -> None:
"""Set a palette colour value from a config string."""
self._guiPalette.setColor(
value, QColor(*self._parseColour(parser, section, name))
@@ -447,14 +445,22 @@ class GuiIcons:
ICON_KEYS = {
# Project and GUI Icons
"novelwriter", "alert_error", "alert_info", "alert_question", "alert_warn",
- "build_excluded", "build_filtered", "build_included", "cls_archive", "cls_character",
- "cls_custom", "cls_entity", "cls_none", "cls_novel", "cls_object", "cls_plot",
- "cls_timeline", "cls_trash", "cls_world", "proj_chapter", "proj_details", "proj_document",
- "proj_folder", "proj_note", "proj_nwx", "proj_section", "proj_scene", "proj_stats",
- "proj_title", "search_cancel", "search_case", "search_loop", "search_preserve",
- "search_project", "search_regex", "search_word", "status_idle", "status_lang",
- "status_lines", "status_stats", "status_time", "view_build", "view_editor", "view_novel",
- "view_outline",
+ "build_excluded", "build_filtered", "build_included", "proj_chapter", "proj_details",
+ "proj_document", "proj_folder", "proj_note", "proj_nwx", "proj_section", "proj_scene",
+ "proj_stats", "proj_title", "status_idle", "status_lang", "status_lines", "status_stats",
+ "status_time", "view_build", "view_editor", "view_novel", "view_outline",
+
+ # Class Icons
+ "cls_archive", "cls_character", "cls_custom", "cls_entity", "cls_none", "cls_novel",
+ "cls_object", "cls_plot", "cls_timeline", "cls_trash", "cls_world",
+
+ # Search Icons
+ "search_cancel", "search_case", "search_loop", "search_preserve", "search_project",
+ "search_regex", "search_word",
+
+ # Format Icons
+ "fmt_bold", "fmt_italic", "fmt_mode-md", "fmt_mode-sc", "fmt_strike", "fmt_subscript",
+ "fmt_superscript", "fmt_underline",
# General Button Icons
"add", "backward", "bookmark", "browse", "checked", "close", "cross", "down", "edit",
@@ -469,21 +475,27 @@ class GuiIcons:
"deco_doc_h0", "deco_doc_h1", "deco_doc_h2", "deco_doc_h3", "deco_doc_h4", "deco_doc_more",
}
+ TOGGLE_ICON_KEYS = {
+ "sticky": ("sticky-on", "sticky-off"),
+ "bullet": ("bullet-on", "bullet-off"),
+ "fmt_mode": ("fmt_mode-sc", "fmt_mode-md"),
+ }
+
IMAGE_MAP = {
"wiz-back": "wizard-back.jpg",
}
- def __init__(self, mainTheme):
+ def __init__(self, mainTheme: GuiTheme) -> None:
self.mainTheme = mainTheme
# Storage
- self._qIcons = {}
- self._themeMap = {}
- self._headerDec = []
- self._confName = "icons.conf"
+ self._qIcons: dict[str, QIcon] = {}
+ self._themeMap: dict[str, Path] = {}
+ self._headerDec: list[QPixmap] = []
# Icon Theme Path
+ self._confName = "icons.conf"
self._iconPath = CONFIG.assetPath("icons")
# Icon Theme Meta
@@ -501,7 +513,7 @@ def __init__(self, mainTheme):
# Actions
##
- def loadTheme(self, iconTheme):
+ def loadTheme(self, iconTheme: str) -> bool:
"""Update the theme map. This is more of an init, since many of
the GUI icons cannot really be replaced without writing specific
update functions for the classes where they're used.
@@ -575,52 +587,60 @@ def loadTheme(self, iconTheme):
# Access Functions
##
- def loadDecoration(self, decoKey, pxW=None, pxH=None):
+ def loadDecoration(self, name: str, w: int | None = None, h: int | None = None) -> QPixmap:
"""Load graphical decoration element based on the decoration
map or the icon map. This function always returns a QPixmap.
"""
- if decoKey in self._themeMap:
- imgPath = self._themeMap[decoKey]
- elif decoKey in self.IMAGE_MAP:
- imgPath = CONFIG.assetPath("images") / self.IMAGE_MAP[decoKey]
+ if name in self._themeMap:
+ imgPath = self._themeMap[name]
+ elif name in self.IMAGE_MAP:
+ imgPath = CONFIG.assetPath("images") / self.IMAGE_MAP[name]
else:
- logger.error("Decoration with name '%s' does not exist", decoKey)
+ logger.error("Decoration with name '%s' does not exist", name)
return QPixmap()
if not imgPath.is_file():
logger.error("Asset not found: %s", imgPath)
return QPixmap()
- theDeco = QPixmap(str(imgPath))
- if pxW is not None and pxH is not None:
- return theDeco.scaled(pxW, pxH, Qt.IgnoreAspectRatio, Qt.SmoothTransformation)
- elif pxW is None and pxH is not None:
- return theDeco.scaledToHeight(pxH, Qt.SmoothTransformation)
- elif pxW is not None and pxH is None:
- return theDeco.scaledToWidth(pxW, Qt.SmoothTransformation)
+ pixmap = QPixmap(str(imgPath))
+ if w is not None and h is not None:
+ return pixmap.scaled(w, h, Qt.IgnoreAspectRatio, Qt.SmoothTransformation)
+ elif w is None and h is not None:
+ return pixmap.scaledToHeight(h, Qt.SmoothTransformation)
+ elif w is not None and h is None:
+ return pixmap.scaledToWidth(w, Qt.SmoothTransformation)
- return theDeco
+ return pixmap
- def getIcon(self, iconKey):
- """Return an icon from the icon buffer. If it doesn't exist,
- return, load it, and if it still doesn't exist, return an empty
- icon.
- """
- if iconKey in self._qIcons:
- return self._qIcons[iconKey]
+ def getIcon(self, name: str) -> QIcon:
+ """Return an icon from the icon buffer, or load it."""
+ if name in self._qIcons:
+ return self._qIcons[name]
else:
- qIcon = self._loadIcon(iconKey)
- self._qIcons[iconKey] = qIcon
- return qIcon
+ icon = self._loadIcon(name)
+ self._qIcons[name] = icon
+ return icon
+
+ def getToggleIcon(self, name: str, size: tuple[int, int]) -> QIcon:
+ """Return a toggle icon from the icon buffer. or load it."""
+ if name in self.TOGGLE_ICON_KEYS:
+ pOne = self.getPixmap(self.TOGGLE_ICON_KEYS[name][0], size)
+ pTwo = self.getPixmap(self.TOGGLE_ICON_KEYS[name][1], size)
+ icon = QIcon()
+ icon.addPixmap(pOne, QIcon.Normal, QIcon.On)
+ icon.addPixmap(pTwo, QIcon.Normal, QIcon.Off)
+ return icon
+ return QIcon()
- def getPixmap(self, iconKey, iconSize):
+ def getPixmap(self, name: str, size: tuple[int, int]) -> QPixmap:
"""Return an icon from the icon buffer as a QPixmap. If it
doesn't exist, return an empty QPixmap.
"""
- qIcon = self.getIcon(iconKey)
- return qIcon.pixmap(iconSize[0], iconSize[1], QIcon.Normal)
+ return self.getIcon(name).pixmap(size[0], size[1], QIcon.Normal)
- def getItemIcon(self, tType, tClass, tLayout, hLevel="H0"):
+ def getItemIcon(self, tType: nwItemType, tClass: nwItemClass,
+ tLayout: nwItemLayout, hLevel: str = "H0") -> QIcon:
"""Get the correct icon for a project item based on type, class
and header level
"""
@@ -647,17 +667,16 @@ def getItemIcon(self, tType, tClass, tLayout, hLevel="H0"):
return self.getIcon(iconName)
- def getHeaderDecoration(self, hLevel):
- """Get the decoration for a specific header level.
- """
+ def getHeaderDecoration(self, hLevel: int) -> QPixmap:
+ """Get the decoration for a specific header level."""
if not self._headerDec:
iPx = self.mainTheme.baseIconSize
self._headerDec = [
- self.loadDecoration("deco_doc_h0", pxH=iPx),
- self.loadDecoration("deco_doc_h1", pxH=iPx),
- self.loadDecoration("deco_doc_h2", pxH=iPx),
- self.loadDecoration("deco_doc_h3", pxH=iPx),
- self.loadDecoration("deco_doc_h4", pxH=iPx),
+ self.loadDecoration("deco_doc_h0", h=iPx),
+ self.loadDecoration("deco_doc_h1", h=iPx),
+ self.loadDecoration("deco_doc_h2", h=iPx),
+ self.loadDecoration("deco_doc_h3", h=iPx),
+ self.loadDecoration("deco_doc_h4", h=iPx),
]
return self._headerDec[minmax(hLevel, 0, 4)]
@@ -665,27 +684,27 @@ def getHeaderDecoration(self, hLevel):
# Internal Functions
##
- def _loadIcon(self, iconKey):
+ def _loadIcon(self, name: str) -> QIcon:
"""Load an icon from the assets themes folder. Is guaranteed to
return a QIcon.
"""
- if iconKey not in self.ICON_KEYS:
- logger.error("Requested unknown icon name '%s'", iconKey)
+ if name not in self.ICON_KEYS:
+ logger.error("Requested unknown icon name '%s'", name)
return QIcon()
# If we just want the app icons, return right away
- if iconKey == "novelwriter":
+ if name == "novelwriter":
return QIcon(str(self._iconPath / "novelwriter.svg"))
- elif iconKey == "proj_nwx":
+ elif name == "proj_nwx":
return QIcon(str(self._iconPath / "x-novelwriter-project.svg"))
# Otherwise, we load from the theme folder
- if iconKey in self._themeMap:
- logger.debug("Loading: %s", self._themeMap[iconKey].name)
- return QIcon(str(self._themeMap[iconKey]))
+ if name in self._themeMap:
+ logger.debug("Loading: %s", self._themeMap[name].name)
+ return QIcon(str(self._themeMap[name]))
# If we didn't find one, give up and return an empty icon
- logger.warning("Did not load an icon for '%s'", iconKey)
+ logger.warning("Did not load an icon for '%s'", name)
return QIcon()
@@ -696,9 +715,8 @@ def _loadIcon(self, iconKey):
# Module Functions
# =============================================================================================== #
-def _loadInternalName(confParser, confFile):
- """Open a conf file and read the 'name' setting.
- """
+def _loadInternalName(confParser: NWConfigParser, confFile: str | Path) -> str:
+ """Open a conf file and read the 'name' setting."""
try:
with open(confFile, mode="r", encoding="utf-8") as inFile:
confParser.read_file(inFile)
diff --git a/novelwriter/guimain.py b/novelwriter/guimain.py
index 4d324677f..413014763 100644
--- a/novelwriter/guimain.py
+++ b/novelwriter/guimain.py
@@ -62,7 +62,7 @@
from novelwriter.core.coretools import ProjectBuilder
from novelwriter.enum import (
- nwDocAction, nwDocMode, nwItemType, nwItemClass, nwWidget, nwView
+ nwDocAction, nwDocInsert, nwDocMode, nwItemType, nwItemClass, nwWidget, nwView
)
from novelwriter.common import getGuiItem, hexToInt
from novelwriter.constants import nwFiles
@@ -243,6 +243,11 @@ def __init__(self) -> None:
SHARED.projectStatusMessage.connect(self.mainStatus.setStatusMessage)
SHARED.spellLanguageChanged.connect(self.mainStatus.setLanguage)
+ self.mainMenu.requestDocAction.connect(self._passDocumentAction)
+ self.mainMenu.requestDocInsert.connect(self._passDocumentInsert)
+ self.mainMenu.requestDocInsertText.connect(self._passDocumentInsert)
+ self.mainMenu.requestDocKeyWordInsert.connect(self.docEditor.insertKeyWord)
+
self.sideBar.viewChangeRequested.connect(self._changeView)
self.projView.selectedItemChanged.connect(self.itemDetails.updateViewBox)
@@ -267,6 +272,8 @@ def __init__(self) -> None:
self.docEditor.novelItemMetaChanged.connect(self.novelView.updateNovelItemMeta)
self.docEditor.statusMessage.connect(self.mainStatus.setStatusMessage)
self.docEditor.spellCheckStateChanged.connect(self.mainMenu.setSpellCheckState)
+ self.docEditor.closeDocumentRequest.connect(self.closeDocEditor)
+ self.docEditor.toggleFocusModeRequest.connect(self.toggleFocusMode)
self.docViewer.loadDocumentTagRequest.connect(self._followTag)
@@ -733,20 +740,6 @@ def importDocument(self) -> bool:
return True
- def passDocumentAction(self, action: nwDocAction) -> None:
- """Pass on document action to the document viewer if it has
- focus, or pass it to the document editor if it or any of
- its child widgets have focus. If neither has focus, ignore the
- action.
- """
- if self.docViewer.hasFocus():
- self.docViewer.docAction(action)
- elif self.docEditor.hasFocus():
- self.docEditor.docAction(action)
- else:
- logger.debug("Action cancelled as neither editor nor viewer has focus")
- return
-
##
# Tree Item Actions
##
@@ -1119,12 +1112,6 @@ def switchFocus(self, paneNo: nwWidget) -> None:
self.outlineView.setTreeFocus()
return
- def closeDocEditor(self) -> None:
- """Close the document editor. This does not hide the editor."""
- self.closeDocument()
- SHARED.project.data.setLastHandle(None, "editor")
- return
-
def closeDocViewer(self, byUser: bool = True) -> bool:
"""Close the document view panel."""
self.docViewer.clearViewer()
@@ -1139,13 +1126,44 @@ def closeDocViewer(self, byUser: bool = True) -> bool:
return not self.splitView.isVisible()
- def toggleFocusMode(self) -> bool:
+ def toggleFullScreenMode(self) -> None:
+ """Toggle full screen mode"""
+ self.setWindowState(self.windowState() ^ Qt.WindowFullScreen)
+ return
+
+ ##
+ # Events
+ ##
+
+ def closeEvent(self, event: QCloseEvent):
+ """Capture the closing event of the GUI and call the close
+ function to handle all the close process steps.
+ """
+ if self.closeMain():
+ event.accept()
+ else:
+ event.ignore()
+ return
+
+ ##
+ # Public Slots
+ ##
+
+ @pyqtSlot()
+ def closeDocEditor(self) -> None:
+ """Close the document editor. This does not hide the editor."""
+ self.closeDocument()
+ SHARED.project.data.setLastHandle(None, "editor")
+ return
+
+ @pyqtSlot()
+ def toggleFocusMode(self) -> None:
"""Handle toggle focus mode. The Main GUI Focus Mode hides tree,
view, statusbar and menu.
"""
if self.docEditor.docHandle is None:
logger.error("No document open, so not activating Focus Mode")
- return False
+ return
self.isFocusMode = not self.isFocusMode
if self.isFocusMode:
@@ -1169,11 +1187,165 @@ def toggleFocusMode(self) -> bool:
elif self.docViewer.docHandle is not None:
self.splitView.setVisible(True)
- return True
+ return
- def toggleFullScreenMode(self) -> None:
- """Toggle full screen mode"""
- self.setWindowState(self.windowState() ^ Qt.WindowFullScreen)
+ ##
+ # Private Slots
+ ##
+
+ @pyqtSlot(str, nwDocMode)
+ def _followTag(self, tag: str, mode: nwDocMode) -> None:
+ """Follow a tag after user interaction with a link."""
+ tHandle, sTitle = self._getTagSource(tag)
+ if tHandle is not None:
+ if mode == nwDocMode.EDIT:
+ self.openDocument(tHandle)
+ elif mode == nwDocMode.VIEW:
+ self.viewDocument(tHandle=tHandle, sTitle=sTitle)
+ return
+
+ @pyqtSlot(str, nwDocMode, str, bool)
+ def _openDocument(self, tHandle: str, mode: nwDocMode, sTitle: str, setFocus: bool) -> None:
+ """Handle an open document request."""
+ if tHandle is not None:
+ if mode == nwDocMode.EDIT:
+ tLine = None
+ hItem = SHARED.project.index.getItemHeader(tHandle, sTitle)
+ if hItem is not None:
+ tLine = hItem.line
+ self.openDocument(tHandle, tLine=tLine, changeFocus=setFocus)
+ elif mode == nwDocMode.VIEW:
+ self.viewDocument(tHandle=tHandle, sTitle=sTitle)
+ return
+
+ @pyqtSlot(nwView)
+ def _changeView(self, view: nwView) -> None:
+ """Handle the requested change of view from the GuiViewBar."""
+ if view == nwView.EDITOR:
+ # Only change the main stack, but not the project stack
+ self.mainStack.setCurrentWidget(self.splitMain)
+
+ elif view == nwView.PROJECT:
+ self.mainStack.setCurrentWidget(self.splitMain)
+ self.projStack.setCurrentWidget(self.projView)
+
+ elif view == nwView.NOVEL:
+ self.mainStack.setCurrentWidget(self.splitMain)
+ self.projStack.setCurrentWidget(self.novelView)
+
+ elif view == nwView.OUTLINE:
+ self.mainStack.setCurrentWidget(self.outlineView)
+
+ return
+
+ @pyqtSlot(nwDocAction)
+ def _passDocumentAction(self, action: nwDocAction) -> None:
+ """Pass on a document action to the document viewer if it has
+ focus, or pass it to the document editor if it or any of its
+ child widgets have focus. If neither has focus, ignore it.
+ """
+ if self.docViewer.hasFocus():
+ self.docViewer.docAction(action)
+ elif self.docEditor.hasFocus():
+ self.docEditor.docAction(action)
+ else:
+ logger.debug("Action cancelled as neither editor nor viewer has focus")
+ return
+
+ @pyqtSlot(str)
+ @pyqtSlot(nwDocInsert)
+ def _passDocumentInsert(self, content: str | nwDocInsert) -> None:
+ """Pass on a document insert action to the document editor if it
+ has focus. If not, ignore it.
+ """
+ if self.docEditor.hasFocus():
+ self.docEditor.insertText(content)
+ return
+
+ @pyqtSlot()
+ def _timeTick(self) -> None:
+ """Process time tick of the main timer."""
+ if not SHARED.hasProject:
+ return
+ currTime = time()
+ editIdle = currTime - self.docEditor.lastActive > CONFIG.userIdleTime
+ userIdle = qApp.applicationState() != Qt.ApplicationActive
+ self.mainStatus.setUserIdle(editIdle or userIdle)
+ SHARED.updateIdleTime(currTime, editIdle or userIdle)
+ self.mainStatus.updateTime(idleTime=SHARED.projectIdleTime)
+ return
+
+ @pyqtSlot()
+ def _autoSaveProject(self) -> None:
+ """Autosave of the project. This is a timer-activated slot."""
+ doSave = SHARED.hasProject
+ doSave &= SHARED.project.projChanged
+ doSave &= SHARED.project.storage.isOpen()
+ if doSave:
+ logger.debug("Autosaving project")
+ self.saveProject(autoSave=True)
+ return
+
+ @pyqtSlot()
+ def _autoSaveDocument(self) -> None:
+ """Autosave of the document. This is a timer-activated slot."""
+ if SHARED.hasProject and self.docEditor.docChanged:
+ logger.debug("Autosaving document")
+ self.saveDocument()
+ return
+
+ @pyqtSlot()
+ def _updateStatusWordCount(self) -> None:
+ """Update the word count on the status bar."""
+ if not SHARED.hasProject:
+ self.mainStatus.setProjectStats(0, 0)
+
+ SHARED.project.updateWordCounts()
+ if CONFIG.incNotesWCount:
+ iTotal = sum(SHARED.project.data.initCounts)
+ cTotal = sum(SHARED.project.data.currCounts)
+ self.mainStatus.setProjectStats(cTotal, cTotal - iTotal)
+ else:
+ iNovel, _ = SHARED.project.data.initCounts
+ cNovel, _ = SHARED.project.data.currCounts
+ self.mainStatus.setProjectStats(cNovel, cNovel - iNovel)
+
+ return
+
+ @pyqtSlot()
+ def _keyPressReturn(self) -> None:
+ """Forward the return/enter keypress to the function that opens
+ the currently selected item.
+ """
+ self.openSelectedItem()
+ return
+
+ @pyqtSlot()
+ def _keyPressEscape(self) -> None:
+ """Process escape keypress in the main window."""
+ if self.docEditor.docSearch.isVisible():
+ self.docEditor.closeSearch()
+ elif self.isFocusMode:
+ self.toggleFocusMode()
+ return
+
+ @pyqtSlot(int)
+ def _mainStackChanged(self, index: int) -> None:
+ """Process main window tab change."""
+ if index == self.idxOutlineView:
+ if SHARED.hasProject:
+ self.outlineView.refreshTree()
+ return
+
+ @pyqtSlot(int)
+ def _projStackChanged(self, index: int) -> None:
+ """Process project view tab change."""
+ sHandle = None
+ if index == self.idxProjView:
+ sHandle = self.projView.getSelectedHandle()
+ elif index == self.idxNovelView:
+ sHandle, _ = self.novelView.getSelectedHandle()
+ self.itemDetails.updateViewBox(sHandle)
return
##
@@ -1328,153 +1500,4 @@ def _getTagSource(self, tag: str) -> tuple[str | None, str | None]:
return None, None
return tHandle, sTitle
- ##
- # Events
- ##
-
- def closeEvent(self, event: QCloseEvent):
- """Capture the closing event of the GUI and call the close
- function to handle all the close process steps.
- """
- if self.closeMain():
- event.accept()
- else:
- event.ignore()
- return
-
- ##
- # Private Slots
- ##
-
- @pyqtSlot(str, nwDocMode)
- def _followTag(self, tag: str, mode: nwDocMode) -> None:
- """Follow a tag after user interaction with a link."""
- tHandle, sTitle = self._getTagSource(tag)
- if tHandle is not None:
- if mode == nwDocMode.EDIT:
- self.openDocument(tHandle)
- elif mode == nwDocMode.VIEW:
- self.viewDocument(tHandle=tHandle, sTitle=sTitle)
- return
-
- @pyqtSlot(str, nwDocMode, str, bool)
- def _openDocument(self, tHandle: str, mode: nwDocMode, sTitle: str, setFocus: bool) -> None:
- """Handle an open document request."""
- if tHandle is not None:
- if mode == nwDocMode.EDIT:
- tLine = None
- hItem = SHARED.project.index.getItemHeader(tHandle, sTitle)
- if hItem is not None:
- tLine = hItem.line
- self.openDocument(tHandle, tLine=tLine, changeFocus=setFocus)
- elif mode == nwDocMode.VIEW:
- self.viewDocument(tHandle=tHandle, sTitle=sTitle)
- return
-
- @pyqtSlot(nwView)
- def _changeView(self, view: nwView) -> None:
- """Handle the requested change of view from the GuiViewBar."""
- if view == nwView.EDITOR:
- # Only change the main stack, but not the project stack
- self.mainStack.setCurrentWidget(self.splitMain)
-
- elif view == nwView.PROJECT:
- self.mainStack.setCurrentWidget(self.splitMain)
- self.projStack.setCurrentWidget(self.projView)
-
- elif view == nwView.NOVEL:
- self.mainStack.setCurrentWidget(self.splitMain)
- self.projStack.setCurrentWidget(self.novelView)
-
- elif view == nwView.OUTLINE:
- self.mainStack.setCurrentWidget(self.outlineView)
-
- return
-
- @pyqtSlot()
- def _timeTick(self) -> None:
- """Process time tick of the main timer."""
- if not SHARED.hasProject:
- return
- currTime = time()
- editIdle = currTime - self.docEditor.lastActive > CONFIG.userIdleTime
- userIdle = qApp.applicationState() != Qt.ApplicationActive
- self.mainStatus.setUserIdle(editIdle or userIdle)
- SHARED.updateIdleTime(currTime, editIdle or userIdle)
- self.mainStatus.updateTime(idleTime=SHARED.projectIdleTime)
- return
-
- @pyqtSlot()
- def _autoSaveProject(self) -> None:
- """Autosave of the project. This is a timer-activated slot."""
- doSave = SHARED.hasProject
- doSave &= SHARED.project.projChanged
- doSave &= SHARED.project.storage.isOpen()
- if doSave:
- logger.debug("Autosaving project")
- self.saveProject(autoSave=True)
- return
-
- @pyqtSlot()
- def _autoSaveDocument(self) -> None:
- """Autosave of the document. This is a timer-activated slot."""
- if SHARED.hasProject and self.docEditor.docChanged:
- logger.debug("Autosaving document")
- self.saveDocument()
- return
-
- @pyqtSlot()
- def _updateStatusWordCount(self) -> None:
- """Update the word count on the status bar."""
- if not SHARED.hasProject:
- self.mainStatus.setProjectStats(0, 0)
-
- SHARED.project.updateWordCounts()
- if CONFIG.incNotesWCount:
- iTotal = sum(SHARED.project.data.initCounts)
- cTotal = sum(SHARED.project.data.currCounts)
- self.mainStatus.setProjectStats(cTotal, cTotal - iTotal)
- else:
- iNovel, _ = SHARED.project.data.initCounts
- cNovel, _ = SHARED.project.data.currCounts
- self.mainStatus.setProjectStats(cNovel, cNovel - iNovel)
-
- return
-
- @pyqtSlot()
- def _keyPressReturn(self) -> None:
- """Forward the return/enter keypress to the function that opens
- the currently selected item.
- """
- self.openSelectedItem()
- return
-
- @pyqtSlot()
- def _keyPressEscape(self) -> None:
- """Process escape keypress in the main window."""
- if self.docEditor.docSearch.isVisible():
- self.docEditor.closeSearch()
- elif self.isFocusMode:
- self.toggleFocusMode()
- return
-
- @pyqtSlot(int)
- def _mainStackChanged(self, index: int) -> None:
- """Process main window tab change."""
- if index == self.idxOutlineView:
- if SHARED.hasProject:
- self.outlineView.refreshTree()
- return
-
- @pyqtSlot(int)
- def _projStackChanged(self, index: int) -> None:
- """Process project view tab change."""
- sHandle = None
- if index == self.idxProjView:
- sHandle = self.projView.getSelectedHandle()
- elif index == self.idxNovelView:
- sHandle, _ = self.novelView.getSelectedHandle()
- self.itemDetails.updateViewBox(sHandle)
- return
-
# END Class GuiMain
diff --git a/tests/reference/baseConfig_novelwriter.conf b/tests/reference/baseConfig_novelwriter.conf
index f3434415f..95df90720 100644
--- a/tests/reference/baseConfig_novelwriter.conf
+++ b/tests/reference/baseConfig_novelwriter.conf
@@ -68,6 +68,8 @@ useridletime = 300
[State]
showrefpanel = True
+showedittoolbar = False
+useshortcodes = False
viewcomments = True
viewsynopsis = True
searchcase = False
diff --git a/tests/reference/guiPreferences_novelwriter.conf b/tests/reference/guiPreferences_novelwriter.conf
index dfa3f3fd5..6ea3a0d53 100644
--- a/tests/reference/guiPreferences_novelwriter.conf
+++ b/tests/reference/guiPreferences_novelwriter.conf
@@ -68,6 +68,8 @@ useridletime = 300
[State]
showrefpanel = True
+showedittoolbar = False
+useshortcodes = False
viewcomments = True
viewsynopsis = True
searchcase = False
diff --git a/tests/test_core/test_core_options.py b/tests/test_core/test_core_options.py
index 747a8f73f..013456302 100644
--- a/tests/test_core/test_core_options.py
+++ b/tests/test_core/test_core_options.py
@@ -140,7 +140,7 @@ def testCoreOptions_SetGet(mockGUI):
assert theOpts.getFloat("GuiProjectDetails", "mockItem", None) is None
assert theOpts.getBool("GuiProjectDetails", "clearDouble", None) is True
assert theOpts.getBool("GuiProjectDetails", "mockItem", None) is None
- assert theOpts.getEnum("GuiNovelView", "lastCol", NovelTreeColumn, None) == nwColHidden
+ assert theOpts.getEnum("GuiNovelView", "lastCol", NovelTreeColumn, nwColHidden) == nwColHidden
# Get from non-existent groups
assert theOpts.getValue("SomeGroup", "mockItem", None) is None
diff --git a/tests/test_gui/test_gui_doceditor.py b/tests/test_gui/test_gui_doceditor.py
index 9a28e5c2b..064fcf17b 100644
--- a/tests/test_gui/test_gui_doceditor.py
+++ b/tests/test_gui/test_gui_doceditor.py
@@ -24,15 +24,15 @@
from tools import C, buildTestProject
from mocked import causeOSError
-from PyQt5.QtCore import QThreadPool, Qt
from PyQt5.QtGui import QTextBlock, QTextCursor, QTextOption
+from PyQt5.QtCore import QThreadPool, Qt
from PyQt5.QtWidgets import QAction, qApp
from novelwriter import CONFIG, SHARED
from novelwriter.enum import nwDocAction, nwDocInsert, nwItemLayout, nwTrinary, nwWidget
from novelwriter.constants import nwKeyWords, nwUnicode
from novelwriter.core.index import countWords
-from novelwriter.gui.doceditor import GuiDocEditor
+from novelwriter.gui.doceditor import GuiDocEditor, GuiDocToolBar
KEY_DELAY = 1
@@ -214,9 +214,8 @@ def testGuiEditor_Actions(qtbot, nwGUI, projPath, ipsumText, mockRnd):
buildTestProject(nwGUI, projPath)
assert nwGUI.openDocument(C.hSceneDoc) is True
- theText = "### A Scene\n\n%s" % "\n\n".join(ipsumText)
- nwGUI.docEditor.replaceText(theText)
-
+ text = "### A Scene\n\n%s" % "\n\n".join(ipsumText)
+ nwGUI.docEditor.replaceText(text)
theDoc = nwGUI.docEditor.document()
# Select/Cut/Copy/Paste/Undo/Redo
@@ -228,7 +227,7 @@ def testGuiEditor_Actions(qtbot, nwGUI, projPath, ipsumText, mockRnd):
assert nwGUI.docEditor.docAction(nwDocAction.SEL_ALL) is True
theCursor = nwGUI.docEditor.textCursor()
assert theCursor.hasSelection() is True
- assert theCursor.selectedText() == theText.replace("\n", "\u2029")
+ assert theCursor.selectedText() == text.replace("\n", "\u2029")
theCursor.clearSelection()
# Select Paragraph
@@ -239,7 +238,7 @@ def testGuiEditor_Actions(qtbot, nwGUI, projPath, ipsumText, mockRnd):
assert theCursor.selectedText() == ipsumText[1]
# Cut Selected Text
- nwGUI.docEditor.replaceText(theText)
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(1000)
assert nwGUI.docEditor.docAction(nwDocAction.SEL_PARA) is True
assert nwGUI.docEditor.docAction(nwDocAction.CUT) is True
@@ -254,10 +253,10 @@ def testGuiEditor_Actions(qtbot, nwGUI, projPath, ipsumText, mockRnd):
# Paste Back In
assert nwGUI.docEditor.docAction(nwDocAction.PASTE) is True
- assert nwGUI.docEditor.getText() == theText
+ assert nwGUI.docEditor.getText() == text
# Copy Next Paragraph
- nwGUI.docEditor.replaceText(theText)
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(1500)
assert nwGUI.docEditor.docAction(nwDocAction.SEL_PARA) is True
assert nwGUI.docEditor.docAction(nwDocAction.COPY) is True
@@ -279,84 +278,132 @@ def testGuiEditor_Actions(qtbot, nwGUI, projPath, ipsumText, mockRnd):
# Emphasis/Undo/Redo
# ==================
- theText = "### A Scene\n\n%s" % ipsumText[0]
- nwGUI.docEditor.replaceText(theText)
+ text = "### A Scene\n\n%s" % ipsumText[0]
+ nwGUI.docEditor.replaceText(text)
# Emphasis
nwGUI.docEditor.setCursorPosition(50)
assert nwGUI.docEditor.docAction(nwDocAction.EMPH) is True
- assert nwGUI.docEditor.getText() == theText.replace("consectetur", "_consectetur_")
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "_consectetur_")
assert nwGUI.docEditor.docAction(nwDocAction.UNDO) is True
- assert nwGUI.docEditor.getText() == theText
+ assert nwGUI.docEditor.getText() == text
# Strong
nwGUI.docEditor.setCursorPosition(50)
assert nwGUI.docEditor.docAction(nwDocAction.STRONG) is True
- assert nwGUI.docEditor.getText() == theText.replace("consectetur", "**consectetur**")
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "**consectetur**")
assert nwGUI.docEditor.docAction(nwDocAction.UNDO) is True
- assert nwGUI.docEditor.getText() == theText
+ assert nwGUI.docEditor.getText() == text
# Strikeout
nwGUI.docEditor.setCursorPosition(50)
assert nwGUI.docEditor.docAction(nwDocAction.STRIKE) is True
- assert nwGUI.docEditor.getText() == theText.replace("consectetur", "~~consectetur~~")
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "~~consectetur~~")
assert nwGUI.docEditor.docAction(nwDocAction.UNDO) is True
- assert nwGUI.docEditor.getText() == theText
+ assert nwGUI.docEditor.getText() == text
# Redo
assert nwGUI.docEditor.docAction(nwDocAction.REDO) is True
- assert nwGUI.docEditor.getText() == theText.replace("consectetur", "~~consectetur~~")
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "~~consectetur~~")
+ assert nwGUI.docEditor.docAction(nwDocAction.UNDO) is True
+ assert nwGUI.docEditor.getText() == text
+
+ # Shortcodes
+ # ==========
+
+ text = "### A Scene\n\n%s" % ipsumText[0]
+ nwGUI.docEditor.replaceText(text)
+
+ # Italic
+ nwGUI.docEditor.setCursorPosition(46)
+ assert nwGUI.docEditor.docAction(nwDocAction.SC_ITALIC) is True
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "[i]consectetur[/i]")
+ assert nwGUI.docEditor.docAction(nwDocAction.UNDO) is True
+ assert nwGUI.docEditor.getText() == text
+
+ # Bold
+ nwGUI.docEditor.setCursorPosition(46)
+ assert nwGUI.docEditor.docAction(nwDocAction.SC_BOLD) is True
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "[b]consectetur[/b]")
+ assert nwGUI.docEditor.docAction(nwDocAction.UNDO) is True
+ assert nwGUI.docEditor.getText() == text
+
+ # Strikethrough
+ nwGUI.docEditor.setCursorPosition(46)
+ assert nwGUI.docEditor.docAction(nwDocAction.SC_STRIKE) is True
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "[s]consectetur[/s]")
+ assert nwGUI.docEditor.docAction(nwDocAction.UNDO) is True
+ assert nwGUI.docEditor.getText() == text
+
+ # Underline
+ nwGUI.docEditor.setCursorPosition(46)
+ assert nwGUI.docEditor.docAction(nwDocAction.SC_ULINE) is True
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "[u]consectetur[/u]")
+ assert nwGUI.docEditor.docAction(nwDocAction.UNDO) is True
+ assert nwGUI.docEditor.getText() == text
+
+ # Superscript
+ nwGUI.docEditor.setCursorPosition(46)
+ assert nwGUI.docEditor.docAction(nwDocAction.SC_SUP) is True
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "[sup]consectetur[/sup]")
+ assert nwGUI.docEditor.docAction(nwDocAction.UNDO) is True
+ assert nwGUI.docEditor.getText() == text
+
+ # Subscript
+ nwGUI.docEditor.setCursorPosition(46)
+ assert nwGUI.docEditor.docAction(nwDocAction.SC_SUB) is True
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "[sub]consectetur[/sub]")
assert nwGUI.docEditor.docAction(nwDocAction.UNDO) is True
- assert nwGUI.docEditor.getText() == theText
+ assert nwGUI.docEditor.getText() == text
# Quotes
# ======
- theText = "### A Scene\n\n%s" % ipsumText[0]
- nwGUI.docEditor.replaceText(theText)
+ text = "### A Scene\n\n%s" % ipsumText[0]
+ nwGUI.docEditor.replaceText(text)
# Add Single Quotes
nwGUI.docEditor.setCursorPosition(50)
assert nwGUI.docEditor.docAction(nwDocAction.S_QUOTE) is True
- assert nwGUI.docEditor.getText() == theText.replace("consectetur", "\u2018consectetur\u2019")
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "\u2018consectetur\u2019")
assert nwGUI.docEditor.docAction(nwDocAction.UNDO) is True
- assert nwGUI.docEditor.getText() == theText
+ assert nwGUI.docEditor.getText() == text
# Add Double Quotes
nwGUI.docEditor.setCursorPosition(50)
assert nwGUI.docEditor.docAction(nwDocAction.D_QUOTE) is True
- assert nwGUI.docEditor.getText() == theText.replace("consectetur", "\u201cconsectetur\u201d")
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "\u201cconsectetur\u201d")
assert nwGUI.docEditor.docAction(nwDocAction.UNDO) is True
- assert nwGUI.docEditor.getText() == theText
+ assert nwGUI.docEditor.getText() == text
# Replace Single Quotes
- repText = theText.replace("consectetur", "'consectetur'")
+ repText = text.replace("consectetur", "'consectetur'")
nwGUI.docEditor.replaceText(repText)
assert nwGUI.docEditor.docAction(nwDocAction.SEL_ALL) is True
assert nwGUI.docEditor.docAction(nwDocAction.REPL_SNG) is True
- assert nwGUI.docEditor.getText() == theText.replace("consectetur", "\u2018consectetur\u2019")
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "\u2018consectetur\u2019")
# Replace Double Quotes
- repText = theText.replace("consectetur", "\"consectetur\"")
+ repText = text.replace("consectetur", "\"consectetur\"")
nwGUI.docEditor.replaceText(repText)
assert nwGUI.docEditor.docAction(nwDocAction.SEL_ALL) is True
assert nwGUI.docEditor.docAction(nwDocAction.REPL_DBL) is True
- assert nwGUI.docEditor.getText() == theText.replace("consectetur", "\u201cconsectetur\u201d")
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "\u201cconsectetur\u201d")
# Remove Line Breaks
# ==================
- theText = "### A Scene\n\n%s" % ipsumText[0]
- repText = theText[:100] + theText[100:].replace(" ", "\n", 3)
+ text = "### A Scene\n\n%s" % ipsumText[0]
+ repText = text[:100] + text[100:].replace(" ", "\n", 3)
nwGUI.docEditor.replaceText(repText)
assert nwGUI.docEditor.docAction(nwDocAction.RM_BREAKS) is True
- assert nwGUI.docEditor.getText().strip() == theText.strip()
+ assert nwGUI.docEditor.getText().strip() == text.strip()
# Format Block
# ============
- theText = "## Scene Title\n\nScene text.\n\n"
- nwGUI.docEditor.replaceText(theText)
+ text = "## Scene Title\n\nScene text.\n\n"
+ nwGUI.docEditor.replaceText(text)
# Header 1
nwGUI.docEditor.setCursorPosition(0)
@@ -437,20 +484,125 @@ def testGuiEditor_Actions(qtbot, nwGUI, projPath, ipsumText, mockRnd):
# END Test testGuiEditor_Actions
+@pytest.mark.gui
+def testGuiEditor_ToolBar(qtbot, nwGUI, projPath, mockRnd):
+ """Test the document actions. This is not an extensive test of the
+ action features, just that the actions are actually called. The
+ various action features are tested when their respective functions
+ are tested.
+ """
+ buildTestProject(nwGUI, projPath)
+ assert nwGUI.openDocument(C.hSceneDoc) is True
+
+ docEditor: GuiDocEditor = nwGUI.docEditor
+ docToolBar: GuiDocToolBar = docEditor.docToolBar
+
+ text = (
+ "### A Scene\n\n"
+ "Text bold one\n\n"
+ "Text bold two\n\n"
+ "Text italic one\n\n"
+ "Text italic two\n\n"
+ "Text strikethrough one\n\n"
+ "Text strikethrough two\n\n"
+ "Text underline one\n\n"
+ "Text superscript one\n\n"
+ "Text subscript one\n\n"
+ )
+ length = len(text)
+ docEditor.replaceText(text)
+ assert len(docEditor.getText()) == length
+
+ # Show the ToolBar
+ assert docToolBar.isVisible() is False
+ docEditor._toggleToolBarVisibility()
+ assert docToolBar.isVisible() is True
+
+ # Markdown Mode
+ assert docToolBar.tbMode.isChecked() is False
+
+ # Click Bold
+ docEditor.setCursorPosition(20)
+ docToolBar.tbBold.click()
+ assert len(docEditor.getText()) == length + 4
+
+ # Click Italic
+ docEditor.setCursorPosition(54)
+ docToolBar.tbItalic.click()
+ assert len(docEditor.getText()) == length + 6
+
+ # Click Strikethrough
+ docEditor.setCursorPosition(90)
+ docToolBar.tbStrike.click()
+ assert len(docEditor.getText()) == length + 10
+
+ # Shortcode Mode
+ docToolBar.tbMode.click()
+ assert docToolBar.tbMode.isChecked() is True
+
+ # Click Bold
+ docEditor.setCursorPosition(39)
+ docToolBar.tbBold.click()
+ assert len(docEditor.getText()) == length + 17
+
+ # Click Italic
+ docEditor.setCursorPosition(80)
+ docToolBar.tbItalic.click()
+ assert len(docEditor.getText()) == length + 24
+
+ # Click Strikethrough
+ docEditor.setCursorPosition(132)
+ docToolBar.tbStrike.click()
+ assert len(docEditor.getText()) == length + 31
+
+ # Click Underline
+ docEditor.setCursorPosition(163)
+ docToolBar.tbUnderline.click()
+ assert len(docEditor.getText()) == length + 38
+
+ # Click Superscript
+ docEditor.setCursorPosition(190)
+ docToolBar.tbSuperscript.click()
+ assert len(docEditor.getText()) == length + 49
+
+ # Click Subscript
+ docEditor.setCursorPosition(223)
+ docToolBar.tbSubscript.click()
+ assert len(docEditor.getText()) == length + 60
+
+ # Check Result
+ assert docEditor.getText() == (
+ "### A Scene\n\n"
+ "Text **bold** one\n\n"
+ "Text [b]bold[/b] two\n\n"
+ "Text _italic_ one\n\n"
+ "Text [i]italic[/i] two\n\n"
+ "Text ~~strikethrough~~ one\n\n"
+ "Text [s]strikethrough[/s] two\n\n"
+ "Text [u]underline[/u] one\n\n"
+ "Text [sup]superscript[/sup] one\n\n"
+ "Text [sub]subscript[/sub] one\n\n"
+ )
+
+ # qtbot.stop()
+
+# END Test testGuiEditor_ToolBar
+
+
@pytest.mark.gui
def testGuiEditor_Insert(qtbot, monkeypatch, nwGUI, projPath, ipsumText, mockRnd):
"""Test the document insert functions."""
buildTestProject(nwGUI, projPath)
assert nwGUI.openDocument(C.hSceneDoc) is True
- theText = "### A Scene\n\n%s" % "\n\n".join(ipsumText)
- nwGUI.docEditor.replaceText(theText)
+ text = "### A Scene\n\n%s" % "\n\n".join(ipsumText)
+ nwGUI.docEditor.replaceText(text)
# Insert Text
# ===========
- theText = "### A Scene\n\n%s" % ipsumText[0]
- nwGUI.docEditor.replaceText(theText)
+ text = "### A Scene\n\n%s" % ipsumText[0]
+ nwGUI.docEditor.replaceText(text)
# No Document Handle
nwGUI.docEditor._docHandle = None
@@ -461,23 +613,23 @@ def testGuiEditor_Insert(qtbot, monkeypatch, nwGUI, projPath, ipsumText, mockRnd
# Insert String
nwGUI.docEditor.setCursorPosition(24)
assert nwGUI.docEditor.insertText(", ipsumer,") is True
- assert nwGUI.docEditor.getText() == theText[:24] + ", ipsumer," + theText[24:]
+ assert nwGUI.docEditor.getText() == text[:24] + ", ipsumer," + text[24:]
# Single Quotes
- nwGUI.docEditor.replaceText(theText)
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(41)
assert nwGUI.docEditor.insertText(nwDocInsert.QUOTE_LS) is True
nwGUI.docEditor.setCursorPosition(53)
assert nwGUI.docEditor.insertText(nwDocInsert.QUOTE_RS) is True
- assert nwGUI.docEditor.getText() == theText.replace("consectetur", "\u2018consectetur\u2019")
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "\u2018consectetur\u2019")
# Double Quotes
- nwGUI.docEditor.replaceText(theText)
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(41)
assert nwGUI.docEditor.insertText(nwDocInsert.QUOTE_LD) is True
nwGUI.docEditor.setCursorPosition(53)
assert nwGUI.docEditor.insertText(nwDocInsert.QUOTE_RD) is True
- assert nwGUI.docEditor.getText() == theText.replace("consectetur", "\u201cconsectetur\u201d")
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "\u201cconsectetur\u201d")
# Invalid Inserts
assert nwGUI.docEditor.insertText(nwDocInsert.NO_INSERT) is False
@@ -486,18 +638,18 @@ def testGuiEditor_Insert(qtbot, monkeypatch, nwGUI, projPath, ipsumText, mockRnd
# Insert KeyWords
# ===============
- theText = "### A Scene\n\n\n%s" % ipsumText[0]
- nwGUI.docEditor.replaceText(theText)
+ text = "### A Scene\n\n\n%s" % ipsumText[0]
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorLine(3)
# Invalid Keyword
assert nwGUI.docEditor.insertKeyWord("stuff") is False
- assert nwGUI.docEditor.getText() == theText
+ assert nwGUI.docEditor.getText() == text
# Valid Keyword
assert nwGUI.docEditor.insertKeyWord(nwKeyWords.POV_KEY) is True
assert nwGUI.docEditor.insertText("Jane\n")
- assert nwGUI.docEditor.getText() == theText.replace(
+ assert nwGUI.docEditor.getText() == text.replace(
"\n\n\n", "\n\n@pov: Jane\n\n", 1
)
@@ -510,7 +662,7 @@ def testGuiEditor_Insert(qtbot, monkeypatch, nwGUI, projPath, ipsumText, mockRnd
nwGUI.docEditor.setCursorPosition(20)
assert nwGUI.docEditor.insertKeyWord(nwKeyWords.CHAR_KEY) is True
assert nwGUI.docEditor.insertText("John")
- assert nwGUI.docEditor.getText() == theText.replace(
+ assert nwGUI.docEditor.getText() == text.replace(
"\n\n\n", "\n\n@pov: Jane\n@char: John\n\n", 1
)
@@ -525,45 +677,45 @@ def testGuiEditor_TextManipulation(qtbot, monkeypatch, nwGUI, projPath, ipsumTex
buildTestProject(nwGUI, projPath)
assert nwGUI.openDocument(C.hSceneDoc) is True
- theText = "### A Scene\n\n%s" % "\n\n".join(ipsumText)
- nwGUI.docEditor.replaceText(theText)
+ text = "### A Scene\n\n%s" % "\n\n".join(ipsumText)
+ nwGUI.docEditor.replaceText(text)
# Clear Surrounding
# =================
# No Selection
- theText = "### A Scene\n\n%s" % ipsumText[0]
- nwGUI.docEditor.replaceText(theText)
+ text = "### A Scene\n\n%s" % ipsumText[0]
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(45)
- theCursor = nwGUI.docEditor.textCursor()
- assert nwGUI.docEditor._clearSurrounding(theCursor, 1) is False
+ cursor = nwGUI.docEditor.textCursor()
+ assert nwGUI.docEditor._clearSurrounding(cursor, 1) is False
# Clear Characters, 1 Layer
- repText = theText.replace("consectetur", "=consectetur=")
+ repText = text.replace("consectetur", "=consectetur=")
nwGUI.docEditor.replaceText(repText)
nwGUI.docEditor.setCursorPosition(45)
- theCursor = nwGUI.docEditor.textCursor()
- theCursor.select(QTextCursor.WordUnderCursor)
- assert nwGUI.docEditor._clearSurrounding(theCursor, 1) is True
- assert nwGUI.docEditor.getText() == theText
+ cursor = nwGUI.docEditor.textCursor()
+ cursor.select(QTextCursor.WordUnderCursor)
+ assert nwGUI.docEditor._clearSurrounding(cursor, 1) is True
+ assert nwGUI.docEditor.getText() == text
# Clear Characters, 2 Layers
- repText = theText.replace("consectetur", "==consectetur==")
+ repText = text.replace("consectetur", "==consectetur==")
nwGUI.docEditor.replaceText(repText)
nwGUI.docEditor.setCursorPosition(45)
- theCursor = nwGUI.docEditor.textCursor()
- theCursor.select(QTextCursor.WordUnderCursor)
- assert nwGUI.docEditor._clearSurrounding(theCursor, 2) is True
- assert nwGUI.docEditor.getText() == theText
+ cursor = nwGUI.docEditor.textCursor()
+ cursor.select(QTextCursor.WordUnderCursor)
+ assert nwGUI.docEditor._clearSurrounding(cursor, 2) is True
+ assert nwGUI.docEditor.getText() == text
# Wrap Selection
# ==============
- theText = "### A Scene\n\n%s" % "\n\n".join(ipsumText[0:2])
- nwGUI.docEditor.replaceText(theText)
+ text = "### A Scene\n\n%s" % "\n\n".join(ipsumText[0:2])
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(45)
# No Selection
@@ -572,23 +724,23 @@ def testGuiEditor_TextManipulation(qtbot, monkeypatch, nwGUI, projPath, ipsumTex
assert nwGUI.docEditor._wrapSelection("=", "=") is False
# Wrap Equal
- nwGUI.docEditor.replaceText(theText)
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(45)
assert nwGUI.docEditor._wrapSelection("=") is True
- assert nwGUI.docEditor.getText() == theText.replace("consectetur", "=consectetur=")
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "=consectetur=")
# Wrap Unequal
- nwGUI.docEditor.replaceText(theText)
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(45)
assert nwGUI.docEditor._wrapSelection("=", "*") is True
- assert nwGUI.docEditor.getText() == theText.replace("consectetur", "=consectetur*")
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "=consectetur*")
# Past Paragraph
- nwGUI.docEditor.replaceText(theText)
- theCursor = nwGUI.docEditor.textCursor()
- theCursor.setPosition(13, QTextCursor.MoveAnchor)
- theCursor.setPosition(1000, QTextCursor.KeepAnchor)
- nwGUI.docEditor.setTextCursor(theCursor)
+ nwGUI.docEditor.replaceText(text)
+ cursor = nwGUI.docEditor.textCursor()
+ cursor.setPosition(13, QTextCursor.MoveAnchor)
+ cursor.setPosition(1000, QTextCursor.KeepAnchor)
+ nwGUI.docEditor.setTextCursor(cursor)
assert nwGUI.docEditor._wrapSelection("=") is True
newText = nwGUI.docEditor.getText()
@@ -599,8 +751,8 @@ def testGuiEditor_TextManipulation(qtbot, monkeypatch, nwGUI, projPath, ipsumTex
# Toggle Format
# =============
- theText = "### A Scene\n\n%s" % "\n\n".join(ipsumText[0:2])
- nwGUI.docEditor.replaceText(theText)
+ text = "### A Scene\n\n%s" % "\n\n".join(ipsumText[0:2])
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(45)
# No Selection
@@ -609,17 +761,17 @@ def testGuiEditor_TextManipulation(qtbot, monkeypatch, nwGUI, projPath, ipsumTex
assert nwGUI.docEditor._toggleFormat(2, "=") is False
# Wrap Single Equal
- nwGUI.docEditor.replaceText(theText)
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(45)
assert nwGUI.docEditor._toggleFormat(1, "=") is True
- assert nwGUI.docEditor.getText() == theText.replace("consectetur", "=consectetur=")
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "=consectetur=")
# Past Paragraph
- nwGUI.docEditor.replaceText(theText)
- theCursor = nwGUI.docEditor.textCursor()
- theCursor.setPosition(13, QTextCursor.MoveAnchor)
- theCursor.setPosition(1000, QTextCursor.KeepAnchor)
- nwGUI.docEditor.setTextCursor(theCursor)
+ nwGUI.docEditor.replaceText(text)
+ cursor = nwGUI.docEditor.textCursor()
+ cursor.setPosition(13, QTextCursor.MoveAnchor)
+ cursor.setPosition(1000, QTextCursor.KeepAnchor)
+ nwGUI.docEditor.setTextCursor(cursor)
assert nwGUI.docEditor._toggleFormat(1, "=") is True
newText = nwGUI.docEditor.getText()
@@ -628,31 +780,31 @@ def testGuiEditor_TextManipulation(qtbot, monkeypatch, nwGUI, projPath, ipsumTex
assert newPara[2] == ipsumText[1]
# Wrap Double Equal
- nwGUI.docEditor.replaceText(theText)
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(45)
assert nwGUI.docEditor._toggleFormat(2, "=") is True
- assert nwGUI.docEditor.getText() == theText.replace("consectetur", "==consectetur==")
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "==consectetur==")
# Toggle Double Equal
- nwGUI.docEditor.replaceText(theText)
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(45)
assert nwGUI.docEditor._toggleFormat(2, "=") is True
assert nwGUI.docEditor._toggleFormat(2, "=") is True
- assert nwGUI.docEditor.getText() == theText
+ assert nwGUI.docEditor.getText() == text
# Toggle Triple+Double Equal
- nwGUI.docEditor.replaceText(theText)
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(45)
assert nwGUI.docEditor._toggleFormat(3, "=") is True
assert nwGUI.docEditor._toggleFormat(2, "=") is True
- assert nwGUI.docEditor.getText() == theText.replace("consectetur", "=consectetur=")
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "=consectetur=")
# Toggle Unequal
- repText = theText.replace("consectetur", "=consectetur==")
+ repText = text.replace("consectetur", "=consectetur==")
nwGUI.docEditor.replaceText(repText)
nwGUI.docEditor.setCursorPosition(45)
assert nwGUI.docEditor._toggleFormat(1, "=") is True
- assert nwGUI.docEditor.getText() == theText.replace("consectetur", "consectetur=")
+ assert nwGUI.docEditor.getText() == text.replace("consectetur", "consectetur=")
assert nwGUI.docEditor._toggleFormat(1, "=") is True
assert nwGUI.docEditor.getText() == repText
@@ -660,15 +812,15 @@ def testGuiEditor_TextManipulation(qtbot, monkeypatch, nwGUI, projPath, ipsumTex
# ==============
# No Selection
- theText = "### A Scene\n\n%s" % ipsumText[0].replace("consectetur", "=consectetur=")
- nwGUI.docEditor.replaceText(theText)
+ text = "### A Scene\n\n%s" % ipsumText[0].replace("consectetur", "=consectetur=")
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(45)
assert nwGUI.docEditor._replaceQuotes("=", "<", ">") is False
# First Paragraph Selected
# This should not replace anything in second paragraph
- theText = "### A Scene\n\n%s" % "\n\n".join(ipsumText[0:2]).replace("ipsum", "=ipsum=")
- nwGUI.docEditor.replaceText(theText)
+ text = "### A Scene\n\n%s" % "\n\n".join(ipsumText[0:2]).replace("ipsum", "=ipsum=")
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(45)
assert nwGUI.docEditor.docAction(nwDocAction.SEL_PARA)
assert nwGUI.docEditor._replaceQuotes("=", "<", ">") is True
@@ -679,12 +831,12 @@ def testGuiEditor_TextManipulation(qtbot, monkeypatch, nwGUI, projPath, ipsumTex
assert newPara[2] == ipsumText[1].replace("ipsum", "=ipsum=")
# Edge of Document
- theText = ipsumText[0].replace("Lorem", "=Lorem=")
- nwGUI.docEditor.replaceText(theText)
+ text = ipsumText[0].replace("Lorem", "=Lorem=")
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(45)
assert nwGUI.docEditor.docAction(nwDocAction.SEL_ALL)
assert nwGUI.docEditor._replaceQuotes("=", "<", ">") is True
- assert nwGUI.docEditor.getText() == theText.replace("=Lorem=", "")
+ assert nwGUI.docEditor.getText() == text.replace("=Lorem=", "")
# Remove Line Breaks
# ==================
@@ -693,20 +845,20 @@ def testGuiEditor_TextManipulation(qtbot, monkeypatch, nwGUI, projPath, ipsumTex
parTwo = ipsumText[1].replace(" ", "\n", 5)
# Remove All
- theText = "### A Scene\n\n%s\n\n%s" % (parOne, parTwo)
- nwGUI.docEditor.replaceText(theText)
+ text = "### A Scene\n\n%s\n\n%s" % (parOne, parTwo)
+ nwGUI.docEditor.replaceText(text)
nwGUI.docEditor.setCursorPosition(45)
nwGUI.docEditor._removeInParLineBreaks()
assert nwGUI.docEditor.getText() == "### A Scene\n\n%s\n" % "\n\n".join(ipsumText[0:2])
# Remove First Paragraph
# Second paragraphs should remain unchanged
- theText = "### A Scene\n\n%s\n\n%s" % (parOne, parTwo)
- nwGUI.docEditor.replaceText(theText)
- theCursor = nwGUI.docEditor.textCursor()
- theCursor.setPosition(16, QTextCursor.MoveAnchor)
- theCursor.setPosition(680, QTextCursor.KeepAnchor)
- nwGUI.docEditor.setTextCursor(theCursor)
+ text = "### A Scene\n\n%s\n\n%s" % (parOne, parTwo)
+ nwGUI.docEditor.replaceText(text)
+ cursor = nwGUI.docEditor.textCursor()
+ cursor.setPosition(16, QTextCursor.MoveAnchor)
+ cursor.setPosition(680, QTextCursor.KeepAnchor)
+ nwGUI.docEditor.setTextCursor(cursor)
nwGUI.docEditor._removeInParLineBreaks()
newText = nwGUI.docEditor.getText()
@@ -720,6 +872,25 @@ def testGuiEditor_TextManipulation(qtbot, monkeypatch, nwGUI, projPath, ipsumTex
assert newPara[6] == twoBits[4]
assert newPara[7] == " ".join(twoBits[5:])
+ # Key Press Events
+ # ================
+ text = "### A Scene\n\n%s\n\n%s" % (parOne, parTwo)
+ nwGUI.docEditor.replaceText(text)
+ assert nwGUI.docEditor.getText() == text
+
+ # Select All
+ qtbot.keyClick(nwGUI.docEditor, Qt.Key_A, modifier=Qt.ControlModifier, delay=KEY_DELAY)
+ qtbot.keyClick(nwGUI.docEditor, Qt.Key_Delete, delay=KEY_DELAY)
+ assert nwGUI.docEditor.getText() == ""
+
+ # Undo
+ qtbot.keyClick(nwGUI.docEditor, Qt.Key_Z, modifier=Qt.ControlModifier, delay=KEY_DELAY)
+ assert nwGUI.docEditor.getText() == text
+
+ # Redo
+ qtbot.keyClick(nwGUI.docEditor, Qt.Key_Y, modifier=Qt.ControlModifier, delay=KEY_DELAY)
+ assert nwGUI.docEditor.getText() == ""
+
# qtbot.stop()
# END Test testGuiEditor_TextManipulation
diff --git a/tests/test_gui/test_gui_guimain.py b/tests/test_gui/test_gui_guimain.py
index 74e0e3522..52e2bc0fd 100644
--- a/tests/test_gui/test_gui_guimain.py
+++ b/tests/test_gui/test_gui_guimain.py
@@ -594,14 +594,18 @@ def testGuiMain_Features(qtbot, nwGUI, projPath, mockRnd):
# ==========
# No document open, so not allowing focus mode
- assert nwGUI.toggleFocusMode() is False
+ nwGUI.toggleFocusMode()
+ assert nwGUI.treePane.isVisible() is True
+ assert nwGUI.mainStatus.isVisible() is True
+ assert nwGUI.mainMenu.isVisible() is True
+ assert nwGUI.sideBar.isVisible() is True
# Open a file in editor and viewer
assert nwGUI.openDocument(C.hSceneDoc)
assert nwGUI.viewDocument(C.hSceneDoc)
# Enable focus mode
- assert nwGUI.toggleFocusMode() is True
+ nwGUI.toggleFocusMode()
assert nwGUI.treePane.isVisible() is False
assert nwGUI.mainStatus.isVisible() is False
assert nwGUI.mainMenu.isVisible() is False
@@ -609,7 +613,7 @@ def testGuiMain_Features(qtbot, nwGUI, projPath, mockRnd):
assert nwGUI.splitView.isVisible() is False
# Disable focus mode
- assert nwGUI.toggleFocusMode() is True
+ nwGUI.toggleFocusMode()
assert nwGUI.treePane.isVisible() is True
assert nwGUI.mainStatus.isVisible() is True
assert nwGUI.mainMenu.isVisible() is True
diff --git a/tests/test_gui/test_gui_theme.py b/tests/test_gui/test_gui_theme.py
index f476ca94d..cfb16ab17 100644
--- a/tests/test_gui/test_gui_theme.py
+++ b/tests/test_gui/test_gui_theme.py
@@ -322,17 +322,17 @@ def testGuiTheme_Icons(qtbot, caplog, monkeypatch, nwGUI, tstPaths):
assert qPix.isNull() is True
# Test image sizes
- qPix = iconCache.loadDecoration("wiz-back", pxW=100, pxH=None)
+ qPix = iconCache.loadDecoration("wiz-back", w=100, h=None)
assert qPix.isNull() is False
assert qPix.width() == 100
assert qPix.height() > 100
- qPix = iconCache.loadDecoration("wiz-back", pxW=None, pxH=100)
+ qPix = iconCache.loadDecoration("wiz-back", w=None, h=100)
assert qPix.isNull() is False
assert qPix.width() < 100
assert qPix.height() == 100
- qPix = iconCache.loadDecoration("wiz-back", pxW=100, pxH=100)
+ qPix = iconCache.loadDecoration("wiz-back", w=100, h=100)
assert qPix.isNull() is False
assert qPix.width() == 100
assert qPix.height() == 100