novelWriter 2.7.1__py3-none-any.whl → 2.7.3__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
novelwriter/__init__.py CHANGED
@@ -49,9 +49,9 @@ __license__ = "GPLv3"
49
49
  __author__ = "Veronica Berglyd Olsen"
50
50
  __maintainer__ = "Veronica Berglyd Olsen"
51
51
  __email__ = "code@vkbo.net"
52
- __version__ = "2.7.1"
53
- __hexversion__ = "0x020701f0"
54
- __date__ = "2025-06-09"
52
+ __version__ = "2.7.3"
53
+ __hexversion__ = "0x020703f0"
54
+ __date__ = "2025-07-07"
55
55
  __status__ = "Stable"
56
56
  __domain__ = "novelwriter.io"
57
57
 
Binary file
Binary file
Binary file
novelwriter/common.py CHANGED
@@ -680,11 +680,12 @@ class NWConfigParser(ConfigParser):
680
680
  """Common: Adapted Config Parser
681
681
 
682
682
  This is a subclass of the standard config parser that adds type safe
683
- helper functions, and support for lists.
683
+ helper functions, and support for lists. It also turns off
684
+ interpolation, which would require % symbols to be escaped (#2455).
684
685
  """
685
686
 
686
687
  def __init__(self) -> None:
687
- super().__init__()
688
+ super().__init__(interpolation=None)
688
689
  return
689
690
 
690
691
  def rdStr(self, section: str, option: str, default: str) -> str:
novelwriter/config.py CHANGED
@@ -44,7 +44,7 @@ from novelwriter.common import (
44
44
  NWConfigParser, checkInt, checkPath, describeFont, fontMatcher,
45
45
  formatTimeStamp, processDialogSymbols, simplified
46
46
  )
47
- from novelwriter.constants import nwFiles, nwHtmlUnicode, nwQuotes, nwUnicode
47
+ from novelwriter.constants import nwFiles, nwQuotes, nwUnicode
48
48
  from novelwriter.error import formatException, logException
49
49
 
50
50
  if TYPE_CHECKING:
@@ -892,7 +892,7 @@ class Config:
892
892
  """
893
893
  self.splashMessage(f"Initialising {kind} font: {font.family()}")
894
894
  metrics = QFontMetrics(font)
895
- for char in nwHtmlUnicode.U_TO_H.keys():
895
+ for char in nwUnicode.UI_SYMBOLS:
896
896
  if not metrics.inFont(char): # type: ignore
897
897
  logger.warning("No glyph U+%04x in font", ord(char)) # pragma: no cover
898
898
  return
novelwriter/constants.py CHANGED
@@ -586,21 +586,14 @@ class nwUnicode:
586
586
  U_TIMES = "\u00d7" # Multiplication sign
587
587
  U_DIVIDE = "\u00f7" # Division sign
588
588
 
589
- # Arrows
590
- U_UTRI = "\u25b2" # Up-pointing triangle
591
- U_UTRIS = "\u25b4" # Up-pointing triangle, small
592
- U_RTRI = "\u25b6" # Right-pointing triangle
593
- U_RTRIS = "\u25b8" # Right-pointing triangle, small
594
- U_DTRI = "\u25bc" # Down-pointing triangle
595
- U_DTRIS = "\u25be" # Down-pointing triangle, small
596
- U_LTRI = "\u25c0" # Left-pointing triangle
597
- U_LTRIS = "\u25c2" # Left-pointing triangle, small
598
-
599
589
  # Special
600
590
  U_UNKN = "\ufffd" # Unknown character
601
591
  U_NAC1 = "\ufffe" # Not a character
602
592
  U_NAC2 = "\uffff" # Not a character
603
593
 
594
+ # Placeholders
595
+ U_LBREAK = "\u21b2" # Downwards Arrow With Tip Leftwards
596
+
604
597
  # HTML Equivalents
605
598
  # ================
606
599
 
@@ -655,15 +648,15 @@ class nwUnicode:
655
648
  H_TIMES = "×"
656
649
  H_DIVIDE = "÷"
657
650
 
658
- # Arrows
659
- H_UTRI = "▲"
660
- H_UTRIS = "▴"
661
- H_RTRI = "▶"
662
- H_RTRIS = "▸"
663
- H_DTRI = "▼"
664
- H_DTRIS = "▾"
665
- H_LTRI = "◀"
666
- H_LTRIS = "◂"
651
+ # Unicode symbols expected to be used on the UI
652
+ UI_SYMBOLS: Final[list[str]] = [
653
+ U_QUOT, U_APOS, U_LAQUO, U_RAQUO, U_LSQUO, U_RSQUO, U_SBQUO, U_SUQUO,
654
+ U_LDQUO, U_RDQUO, U_BDQUO, U_UDQUO, U_LSAQUO, U_RSAQUO, U_BDRQUO,
655
+ U_LCQUO, U_RCQUO, U_LWCQUO, U_RWCQUO, U_FGDASH, U_ENDASH, U_EMDASH,
656
+ U_HBAR, U_HELLIP, U_MAPOS, U_PRIME, U_DPRIME, U_NBSP, U_THSP, U_THNBSP,
657
+ U_ENSP, U_EMSP, U_MMSP, U_CHECK, U_CROSS, U_BULL, U_TRBULL, U_HYBULL,
658
+ U_FLOWER, U_PERMIL, U_DEGREE, U_MINUS, U_TIMES, U_DIVIDE, U_LBREAK,
659
+ ]
667
660
 
668
661
 
669
662
  class nwHtmlUnicode:
novelwriter/core/index.py CHANGED
@@ -708,17 +708,19 @@ class Index:
708
708
  return 0, 0, 0
709
709
 
710
710
  def getReferences(self, tHandle: str, sTitle: str | None = None) -> dict[str, list[str]]:
711
- """Extract all references made in a file, and optionally title
712
- section.
711
+ """Extract all tags and references made in a file, and
712
+ optionally title section.
713
713
  """
714
- tRefs = {x: [] for x in nwKeyWords.VALID_KEYS}
714
+ refs = {x: [] for x in nwKeyWords.VALID_KEYS}
715
715
  for rTitle, hItem in self._itemIndex.iterItemHeaders(tHandle):
716
716
  if sTitle is None or sTitle == rTitle:
717
717
  for aTag, refTypes in hItem.references.items():
718
718
  for refType in refTypes:
719
- if refType in tRefs:
720
- tRefs[refType].append(self._tagsIndex.tagName(aTag))
721
- return tRefs
719
+ if refType in refs:
720
+ refs[refType].append(self._tagsIndex.tagName(aTag))
721
+ if tag := hItem.tag:
722
+ refs[nwKeyWords.TAG_KEY] = [self._tagsIndex.tagName(tag)]
723
+ return refs
722
724
 
723
725
  def getReferenceForHeader(self, tHandle: str, nHead: int, keyClass: str) -> list[str]:
724
726
  """Get the display names for a tags class for insertion into a
@@ -342,20 +342,26 @@ class IndexHeading:
342
342
  ##
343
343
 
344
344
  def getReferences(self) -> dict[str, list[str]]:
345
- """Extract all references for this heading."""
345
+ """Extract all tags and references for this heading."""
346
346
  refs = {x: [] for x in nwKeyWords.VALID_KEYS}
347
347
  for tag, types in self._refs.items():
348
348
  for keyword in types:
349
349
  if keyword in refs and (name := self._cache.tags.tagName(tag)):
350
350
  refs[keyword].append(name)
351
+ if name := self._cache.tags.tagName(self._tag):
352
+ refs[nwKeyWords.TAG_KEY] = [name]
351
353
  return refs
352
354
 
353
355
  def getReferencesByKeyword(self, keyword: str) -> list[str]:
354
356
  """Extract all references for this heading."""
355
357
  refs = []
356
- for tag, types in self._refs.items():
357
- if keyword in types and (name := self._cache.tags.tagName(tag)):
358
+ if keyword == nwKeyWords.TAG_KEY:
359
+ if name := self._cache.tags.tagName(self._tag):
358
360
  refs.append(name)
361
+ else:
362
+ for tag, types in self._refs.items():
363
+ if keyword in types and (name := self._cache.tags.tagName(tag)):
364
+ refs.append(name)
359
365
  return refs
360
366
 
361
367
  ##
@@ -29,7 +29,7 @@ import logging
29
29
  from PyQt6.QtCore import Qt, pyqtSignal, pyqtSlot
30
30
  from PyQt6.QtGui import QAction, QCloseEvent, QKeyEvent, QKeySequence
31
31
  from PyQt6.QtWidgets import (
32
- QCompleter, QDialogButtonBox, QFileDialog, QHBoxLayout, QLineEdit,
32
+ QCompleter, QDialogButtonBox, QFileDialog, QHBoxLayout, QLineEdit, QMenu,
33
33
  QPushButton, QVBoxLayout, QWidget
34
34
  )
35
35
 
@@ -600,6 +600,7 @@ class GuiPreferences(NDialog):
600
600
  self.sidebar.addButton(title, section)
601
601
  self.mainForm.addGroupLabel(title, section)
602
602
 
603
+ # Dialogue Quotes
603
604
  self.dialogStyle = NComboBox(self)
604
605
  self.dialogStyle.addItem(self.tr("None"), 0)
605
606
  self.dialogStyle.addItem(self.tr("Single Quotes"), 1)
@@ -611,6 +612,15 @@ class GuiPreferences(NDialog):
611
612
  self.tr("Applies to the selected quote styles.")
612
613
  )
613
614
 
615
+ # Open-Ended Dialogue
616
+ self.allowOpenDial = NSwitch(self)
617
+ self.allowOpenDial.setChecked(CONFIG.allowOpenDial)
618
+ self.mainForm.addRow(
619
+ self.tr("Allow open-ended dialogue"), self.allowOpenDial,
620
+ self.tr("Highlight dialogue line with no closing quote.")
621
+ )
622
+
623
+ # Alternative Dialogue
614
624
  self.altDialogOpen = QLineEdit(self)
615
625
  self.altDialogOpen.setMaxLength(4)
616
626
  self.altDialogOpen.setFixedWidth(boxFixed)
@@ -628,23 +638,30 @@ class GuiPreferences(NDialog):
628
638
  self.tr("Custom highlighting of dialogue text.")
629
639
  )
630
640
 
631
- self.allowOpenDial = NSwitch(self)
632
- self.allowOpenDial.setChecked(CONFIG.allowOpenDial)
633
- self.mainForm.addRow(
634
- self.tr("Allow open-ended dialogue"), self.allowOpenDial,
635
- self.tr("Highlight dialogue line with no closing quote.")
636
- )
641
+ # Dialogue Line
642
+ self.mnLineSymbols = QMenu(self)
643
+ for symbol in nwQuotes.ALLOWED:
644
+ label = trConst(nwQuotes.SYMBOLS.get(symbol, nwQuotes.DASHES.get(symbol, "None")))
645
+ self.mnLineSymbols.addAction(
646
+ f"[ {symbol } ] {label}",
647
+ lambda symbol=symbol: self._insertDialogLineSymbol(symbol)
648
+ )
637
649
 
638
650
  self.dialogLine = QLineEdit(self)
639
- self.dialogLine.setMaxLength(4)
640
- self.dialogLine.setFixedWidth(boxFixed)
651
+ self.dialogLine.setMinimumWidth(100)
641
652
  self.dialogLine.setAlignment(QtAlignCenter)
642
- self.dialogLine.setText(CONFIG.dialogLine)
653
+ self.dialogLine.setText(" ".join(CONFIG.dialogLine))
654
+
655
+ self.dialogLineButton = NIconToolButton(self, iSz, "add", "green")
656
+ self.dialogLineButton.setMenu(self.mnLineSymbols)
657
+
643
658
  self.mainForm.addRow(
644
659
  self.tr("Dialogue line symbols"), self.dialogLine,
645
- self.tr("Lines starting with any of these symbols are dialogue.")
660
+ self.tr("Lines starting with any of these symbols are dialogue."),
661
+ button=self.dialogLineButton
646
662
  )
647
663
 
664
+ # Narrator Break
648
665
  self.narratorBreak = NComboBox(self)
649
666
  self.narratorDialog = NComboBox(self)
650
667
  for key, value in nwQuotes.DASHES.items():
@@ -664,6 +681,7 @@ class GuiPreferences(NDialog):
664
681
  self.tr("Alternates dialogue highlighting within any paragraph.")
665
682
  )
666
683
 
684
+ # Emphasis
667
685
  self.highlightEmph = NSwitch(self)
668
686
  self.highlightEmph.setChecked(CONFIG.highlightEmph)
669
687
  self.mainForm.addRow(
@@ -671,6 +689,7 @@ class GuiPreferences(NDialog):
671
689
  self.tr("Applies to the document editor only.")
672
690
  )
673
691
 
692
+ # Additional Spaces
674
693
  self.showMultiSpaces = NSwitch(self)
675
694
  self.showMultiSpaces.setChecked(CONFIG.showMultiSpaces)
676
695
  self.mainForm.addRow(
@@ -906,6 +925,14 @@ class GuiPreferences(NDialog):
906
925
  self.askBeforeBackup.setEnabled(state)
907
926
  return
908
927
 
928
+ @pyqtSlot(str)
929
+ def _insertDialogLineSymbol(self, symbol: str) -> None:
930
+ """Insert a symbol in the dialogue line box."""
931
+ current = self.dialogLine.text()
932
+ values = processDialogSymbols(f"{current} {symbol}")
933
+ self.dialogLine.setText(" ".join(values))
934
+ return
935
+
909
936
  @pyqtSlot(bool)
910
937
  def _toggleAutoReplaceMain(self, state: bool) -> None:
911
938
  """Toggle switches controlled by the auto replace switch."""
@@ -37,7 +37,7 @@ from PyQt6.QtCore import QMargins, QSize
37
37
  from novelwriter import __version__
38
38
  from novelwriter.common import firstFloat, xmlElement, xmlSubElem
39
39
  from novelwriter.constants import nwHeadFmt, nwStyles
40
- from novelwriter.formats.shared import BlockFmt, BlockTyp, T_Formats, TextFmt
40
+ from novelwriter.formats.shared import BlockFmt, BlockTyp, T_Formats, TextFmt, stripEscape
41
41
  from novelwriter.formats.tokenizer import Tokenizer
42
42
  from novelwriter.types import QtHexRgb
43
43
 
@@ -480,7 +480,7 @@ class ToDocX(Tokenizer):
480
480
  xmlSubElem(rPr, _wTag("color"), attrib={W_VAL: _docXCol(color)})
481
481
 
482
482
  if isinstance(text, str):
483
- for segment in RX_TEXT.split(text):
483
+ for segment in RX_TEXT.split(stripEscape(text)):
484
484
  if segment == "\n":
485
485
  xmlSubElem(xR, _wTag("br"))
486
486
  elif segment == "\t":
@@ -495,12 +495,10 @@ class Tokenizer(ABC):
495
495
  def doPreProcessing(self) -> None:
496
496
  """Run pre-processing jobs before the text is tokenized."""
497
497
  # Process the user's auto-replace dictionary
498
- if autoReplace := self._project.data.autoReplace:
499
- repDict = {}
500
- for aKey, aVal in autoReplace.items():
501
- repDict[f"<{aKey}>"] = aVal
502
- xRep = re.compile("|".join([re.escape(k) for k in repDict.keys()]), flags=re.DOTALL)
503
- self._text = xRep.sub(lambda x: repDict[x.group(0)], self._text)
498
+ if entry := self._project.data.autoReplace:
499
+ replace = {f"<{k}>": v for k, v in entry.items()}
500
+ rxRep = re.compile("|".join([re.escape(k) for k in replace]), flags=re.DOTALL)
501
+ self._text = rxRep.sub(lambda x: replace[x.group(0)], self._text)
504
502
  return
505
503
 
506
504
  def tokenizeText(self) -> None:
@@ -882,31 +880,21 @@ class Tokenizer(ABC):
882
880
  if nBlock[0] != BlockTyp.TEXT:
883
881
  # Next block is not text, so we add the buffer to blocks
884
882
  nLines = len(pLines)
885
- cStyle = pLines[0][4]
886
- if firstIndent and not (self._noIndent or cStyle & BlockFmt.ALIGNED):
887
- # If paragraph indentation is enabled, not temporarily
888
- # turned off, and the block is not aligned, we add the
889
- # text indentation flag
890
- cStyle |= BlockFmt.IND_T
883
+ tFmt: T_Formats = []
884
+ pTxt = ""
885
+ cStyle = BlockFmt.NONE
891
886
 
892
887
  if nLines == 1:
893
- # The paragraph contains a single line, so we just save
894
- # that directly to the blocks list. If justify is
895
- # enabled, and there is no alignment, we apply it.
896
- if doJustify and not cStyle & BlockFmt.ALIGNED:
897
- cStyle |= BlockFmt.JUSTIFY
898
-
888
+ # The paragraph contains a single line
889
+ tFmt = pLines[0][3]
899
890
  pTxt = pLines[0][2].translate(transMapB)
900
- sBlocks.append((
901
- BlockTyp.TEXT, pLines[0][1], pTxt, pLines[0][3], cStyle
902
- ))
891
+ cStyle = pLines[0][4]
903
892
 
904
893
  elif nLines > 1:
905
894
  # The paragraph contains multiple lines, so we need to
906
895
  # join them according to the line break policy, and
907
896
  # recompute all the formatting markers
908
897
  tTxt = ""
909
- tFmt: T_Formats = []
910
898
  for aBlock in pLines:
911
899
  tLen = len(tTxt)
912
900
  tTxt += f"{aBlock[2]}{lineSep}"
@@ -914,6 +902,18 @@ class Tokenizer(ABC):
914
902
  cStyle |= aBlock[4]
915
903
 
916
904
  pTxt = tTxt[:-1].translate(transMapB)
905
+
906
+ if nLines:
907
+ isAligned = cStyle & BlockFmt.ALIGNED
908
+ if firstIndent and not (self._noIndent or isAligned):
909
+ # If paragraph indentation is enabled, not temporarily
910
+ # turned off, and the block is not aligned, we add the
911
+ # text indentation flag
912
+ cStyle |= BlockFmt.IND_T
913
+
914
+ if doJustify and not isAligned:
915
+ cStyle |= BlockFmt.JUSTIFY
916
+
917
917
  sBlocks.append((
918
918
  BlockTyp.TEXT, pLines[0][1], pTxt, tFmt, cStyle
919
919
  ))
@@ -1170,12 +1170,18 @@ class Tokenizer(ABC):
1170
1170
 
1171
1171
  class HeadingFormatter:
1172
1172
 
1173
- def __init__(self, project: NWProject) -> None:
1173
+ def __init__(
1174
+ self,
1175
+ project: NWProject,
1176
+ chapter: int = 0,
1177
+ scene: int = 0,
1178
+ absolute: int = 0,
1179
+ ) -> None:
1174
1180
  self._project = project
1175
1181
  self._handle = None
1176
- self._chCount = 0
1177
- self._scChCount = 0
1178
- self._scAbsCount = 0
1182
+ self._chapter = chapter
1183
+ self._scene = scene
1184
+ self._absolute = absolute
1179
1185
  return
1180
1186
 
1181
1187
  def setHandle(self, tHandle: str | None) -> None:
@@ -1185,42 +1191,42 @@ class HeadingFormatter:
1185
1191
 
1186
1192
  def incChapter(self) -> None:
1187
1193
  """Increment the chapter counter."""
1188
- self._chCount += 1
1194
+ self._chapter += 1
1189
1195
  return
1190
1196
 
1191
1197
  def incScene(self) -> None:
1192
1198
  """Increment the scene counters."""
1193
- self._scChCount += 1
1194
- self._scAbsCount += 1
1199
+ self._scene += 1
1200
+ self._absolute += 1
1195
1201
  return
1196
1202
 
1197
1203
  def resetAll(self) -> None:
1198
1204
  """Reset all counters."""
1199
- self._chCount = 0
1200
- self._scChCount = 0
1201
- self._scAbsCount = 0
1205
+ self._chapter = 0
1206
+ self._scene = 0
1207
+ self._absolute = 0
1202
1208
  return
1203
1209
 
1204
1210
  def resetScene(self) -> None:
1205
1211
  """Reset the chapter scene counter."""
1206
- self._scChCount = 0
1212
+ self._scene = 0
1207
1213
  return
1208
1214
 
1209
1215
  def apply(self, hFormat: str, text: str, nHead: int) -> str:
1210
1216
  """Apply formatting to a specific heading."""
1211
1217
  hFormat = hFormat.replace(nwHeadFmt.TITLE, text)
1212
1218
  hFormat = hFormat.replace(nwHeadFmt.BR, "\n")
1213
- hFormat = hFormat.replace(nwHeadFmt.CH_NUM, str(self._chCount))
1214
- hFormat = hFormat.replace(nwHeadFmt.SC_NUM, str(self._scChCount))
1215
- hFormat = hFormat.replace(nwHeadFmt.SC_ABS, str(self._scAbsCount))
1219
+ hFormat = hFormat.replace(nwHeadFmt.CH_NUM, str(self._chapter))
1220
+ hFormat = hFormat.replace(nwHeadFmt.SC_NUM, str(self._scene))
1221
+ hFormat = hFormat.replace(nwHeadFmt.SC_ABS, str(self._absolute))
1216
1222
  if nwHeadFmt.CH_WORD in hFormat:
1217
- chWord = self._project.localLookup(self._chCount)
1223
+ chWord = self._project.localLookup(self._chapter)
1218
1224
  hFormat = hFormat.replace(nwHeadFmt.CH_WORD, chWord)
1219
1225
  if nwHeadFmt.CH_ROML in hFormat:
1220
- chRom = numberToRoman(self._chCount, toLower=True)
1226
+ chRom = numberToRoman(self._chapter, toLower=True)
1221
1227
  hFormat = hFormat.replace(nwHeadFmt.CH_ROML, chRom)
1222
1228
  if nwHeadFmt.CH_ROMU in hFormat:
1223
- chRom = numberToRoman(self._chCount, toLower=False)
1229
+ chRom = numberToRoman(self._chapter, toLower=False)
1224
1230
  hFormat = hFormat.replace(nwHeadFmt.CH_ROMU, chRom)
1225
1231
 
1226
1232
  if nwHeadFmt.CHAR_POV in hFormat or nwHeadFmt.CHAR_FOCUS in hFormat:
@@ -36,12 +36,13 @@ from PyQt6.QtPrintSupport import QPrinter
36
36
 
37
37
  from novelwriter import __version__
38
38
  from novelwriter.constants import nwStyles, nwUnicode
39
- from novelwriter.formats.shared import BlockFmt, BlockTyp, T_Formats, TextFmt
39
+ from novelwriter.formats.shared import BlockFmt, BlockTyp, T_Formats, TextFmt, stripEscape
40
40
  from novelwriter.formats.tokenizer import HEADINGS, Tokenizer
41
41
  from novelwriter.types import (
42
42
  QtAlignAbsolute, QtAlignCenter, QtAlignJustify, QtAlignLeft, QtAlignRight,
43
- QtKeepAnchor, QtMoveAnchor, QtPageBreakAfter, QtPageBreakBefore,
44
- QtPropLineHeight, QtTransparent, QtVAlignNormal, QtVAlignSub, QtVAlignSuper
43
+ QtKeepAnchor, QtMoveAnchor, QtPageBreakAfter, QtPageBreakAuto,
44
+ QtPageBreakBefore, QtPropLineHeight, QtTransparent, QtVAlignNormal,
45
+ QtVAlignSub, QtVAlignSuper
45
46
  )
46
47
 
47
48
  if TYPE_CHECKING:
@@ -253,8 +254,10 @@ class ToQTextDocument(Tokenizer):
253
254
 
254
255
  elif tType in HEADINGS:
255
256
  bFmt, cFmt = self._genHeadStyle(tType, tMeta, bFmt)
256
- newBlock(cursor, bFmt)
257
- cursor.insertText(tText, cFmt)
257
+ for tPart in tText.split("\n"):
258
+ newBlock(cursor, bFmt)
259
+ cursor.insertText(tPart, cFmt)
260
+ bFmt.setPageBreakPolicy(QtPageBreakAuto)
258
261
 
259
262
  elif tType == BlockTyp.SEP:
260
263
  newBlock(cursor, bFmt)
@@ -343,7 +346,7 @@ class ToQTextDocument(Tokenizer):
343
346
  for pos, fmt, data in tFmt:
344
347
 
345
348
  # Insert buffer with previous format
346
- cursor.insertText(temp[start:pos], cFmt)
349
+ cursor.insertText(stripEscape(temp[start:pos]), cFmt)
347
350
 
348
351
  # Construct next format
349
352
  if fmt == TextFmt.B_B:
@@ -425,13 +428,12 @@ class ToQTextDocument(Tokenizer):
425
428
  if field := data.partition(":")[2]:
426
429
  self._usedFields.append((cursor.position(), field))
427
430
  cursor.insertText("0", cFmt)
428
- pass
429
431
 
430
432
  # Move pos for next pass
431
433
  start = pos
432
434
 
433
435
  # Insert whatever is left in the buffer
434
- cursor.insertText(temp[start:], cFmt)
436
+ cursor.insertText(stripEscape(temp[start:]), cFmt)
435
437
 
436
438
  return
437
439
 
@@ -228,6 +228,8 @@ class GuiDocViewer(QTextBrowser):
228
228
  qDoc.setTheme(self._docTheme)
229
229
  qDoc.initDocument()
230
230
  qDoc.setKeywords(True)
231
+ qDoc.setCommentType(nwComment.NOTE, CONFIG.viewComments)
232
+ qDoc.setCommentType(nwComment.STORY, CONFIG.viewComments)
231
233
  qDoc.setCommentType(nwComment.PLAIN, CONFIG.viewComments)
232
234
  qDoc.setCommentType(nwComment.SYNOPSIS, CONFIG.viewSynopsis)
233
235
  qDoc.setCommentType(nwComment.SHORT, CONFIG.viewSynopsis)
@@ -593,6 +593,7 @@ class GuiNovelTree(NTreeView):
593
593
  lines = []
594
594
  if head := SHARED.project.index.getItemHeading(tHandle, sTitle):
595
595
  tags = head.getReferences()
596
+ appendTags(tags, nwKeyWords.TAG_KEY, lines)
596
597
  appendTags(tags, nwKeyWords.POV_KEY, lines)
597
598
  appendTags(tags, nwKeyWords.FOCUS_KEY, lines)
598
599
  appendTags(tags, nwKeyWords.CHAR_KEY, lines)
@@ -43,7 +43,7 @@ from PyQt6.QtWidgets import (
43
43
 
44
44
  from novelwriter import CONFIG, SHARED
45
45
  from novelwriter.common import fuzzyTime, qtLambda
46
- from novelwriter.constants import nwLabels, nwStats, trStats
46
+ from novelwriter.constants import nwHeadFmt, nwLabels, nwStats, nwUnicode, trStats
47
47
  from novelwriter.core.buildsettings import BuildCollection, BuildSettings
48
48
  from novelwriter.core.docbuild import NWBuildDocument
49
49
  from novelwriter.extensions.modified import NIconToggleButton, NIconToolButton, NToolDialog
@@ -595,11 +595,7 @@ class _DetailsWidget(QWidget):
595
595
  item.addChild(sub)
596
596
 
597
597
  # Headings
598
- hFmt = HeadingFormatter(SHARED.project)
599
- hFmt.incChapter()
600
- hFmt.incScene()
601
- hFmt.resetScene()
602
- hFmt.incScene()
598
+ hFmt = HeadingFormatter(SHARED.project, 7, 5, 23)
603
599
  title = self.tr("Title")
604
600
  hidden = self.tr("Hidden")
605
601
 
@@ -620,7 +616,8 @@ class _DetailsWidget(QWidget):
620
616
  if build.getBool(hHide):
621
617
  sub.setText(1, f"[{hidden}]")
622
618
  else:
623
- sub.setText(1, hFmt.apply(build.getStr(hFormat), title, 0))
619
+ preview = build.getStr(hFormat).replace(nwHeadFmt.BR, nwUnicode.U_LBREAK)
620
+ sub.setText(1, hFmt.apply(preview, title, 0))
624
621
  item.addChild(sub)
625
622
 
626
623
  # Text Content
@@ -38,7 +38,7 @@ from PyQt6.QtWidgets import (
38
38
 
39
39
  from novelwriter import CONFIG, SHARED
40
40
  from novelwriter.common import describeFont, fontMatcher, qtAddAction, qtLambda
41
- from novelwriter.constants import nwHeadFmt, nwKeyWords, nwLabels, trConst
41
+ from novelwriter.constants import nwHeadFmt, nwKeyWords, nwLabels, nwUnicode, trConst
42
42
  from novelwriter.core.buildsettings import BuildSettings, FilterMode
43
43
  from novelwriter.extensions.configlayout import (
44
44
  NColorLabel, NFixedPage, NScrollableForm, NScrollablePage
@@ -795,12 +795,15 @@ class _HeadingsTab(NScrollablePage):
795
795
 
796
796
  def loadContent(self) -> None:
797
797
  """Populate the widgets."""
798
- self.fmtPart.setText(self._build.getStr("headings.fmtPart"))
799
- self.fmtChapter.setText(self._build.getStr("headings.fmtChapter"))
800
- self.fmtUnnumbered.setText(self._build.getStr("headings.fmtUnnumbered"))
801
- self.fmtScene.setText(self._build.getStr("headings.fmtScene"))
802
- self.fmtAScene.setText(self._build.getStr("headings.fmtAltScene"))
803
- self.fmtSection.setText(self._build.getStr("headings.fmtSection"))
798
+ def fmtBreak(text: str) -> str:
799
+ return text.replace(nwHeadFmt.BR, nwUnicode.U_LBREAK)
800
+
801
+ self.fmtPart.setText(fmtBreak(self._build.getStr("headings.fmtPart")))
802
+ self.fmtChapter.setText(fmtBreak(self._build.getStr("headings.fmtChapter")))
803
+ self.fmtUnnumbered.setText(fmtBreak(self._build.getStr("headings.fmtUnnumbered")))
804
+ self.fmtScene.setText(fmtBreak(self._build.getStr("headings.fmtScene")))
805
+ self.fmtAScene.setText(fmtBreak(self._build.getStr("headings.fmtAltScene")))
806
+ self.fmtSection.setText(fmtBreak(self._build.getStr("headings.fmtSection")))
804
807
 
805
808
  self.swtPart.setChecked(self._build.getBool("headings.hidePart"))
806
809
  self.swtChapter.setChecked(self._build.getBool("headings.hideChapter"))
@@ -880,7 +883,7 @@ class _HeadingsTab(NScrollablePage):
880
883
  text = ""
881
884
  label = self.tr("None")
882
885
 
883
- self.editTextBox.setPlainText(text.replace(nwHeadFmt.BR, "\n"))
886
+ self.editTextBox.setPlainText(text.replace(nwUnicode.U_LBREAK, "\n"))
884
887
  self.lblEditForm.setText(self.tr("Editing: {0}").format(label))
885
888
 
886
889
  return
@@ -893,25 +896,26 @@ class _HeadingsTab(NScrollablePage):
893
896
  def _saveFormat(self) -> None:
894
897
  """Save the format from the edit text box."""
895
898
  heading = self._editing
896
- text = self.editTextBox.toPlainText().strip().replace("\n", nwHeadFmt.BR)
899
+ text = self.editTextBox.toPlainText().strip().replace("\n", nwUnicode.U_LBREAK)
900
+ value = text.replace(nwUnicode.U_LBREAK, nwHeadFmt.BR)
897
901
  if heading == self.EDIT_TITLE:
898
902
  self.fmtPart.setText(text)
899
- self._build.setValue("headings.fmtPart", text)
903
+ self._build.setValue("headings.fmtPart", value)
900
904
  elif heading == self.EDIT_CHAPTER:
901
905
  self.fmtChapter.setText(text)
902
- self._build.setValue("headings.fmtChapter", text)
906
+ self._build.setValue("headings.fmtChapter", value)
903
907
  elif heading == self.EDIT_UNNUM:
904
908
  self.fmtUnnumbered.setText(text)
905
- self._build.setValue("headings.fmtUnnumbered", text)
909
+ self._build.setValue("headings.fmtUnnumbered", value)
906
910
  elif heading == self.EDIT_SCENE:
907
911
  self.fmtScene.setText(text)
908
- self._build.setValue("headings.fmtScene", text)
912
+ self._build.setValue("headings.fmtScene", value)
909
913
  elif heading == self.EDIT_HSCENE:
910
914
  self.fmtAScene.setText(text)
911
- self._build.setValue("headings.fmtAltScene", text)
915
+ self._build.setValue("headings.fmtAltScene", value)
912
916
  elif heading == self.EDIT_SECTION:
913
917
  self.fmtSection.setText(text)
914
- self._build.setValue("headings.fmtSection", text)
918
+ self._build.setValue("headings.fmtSection", value)
915
919
  else:
916
920
  return
917
921
 
novelwriter/types.py CHANGED
@@ -52,6 +52,7 @@ QtVAlignSuper = QTextCharFormat.VerticalAlignment.AlignSuperScript
52
52
 
53
53
  QtPageBreakBefore = QTextFormat.PageBreakFlag.PageBreak_AlwaysBefore
54
54
  QtPageBreakAfter = QTextFormat.PageBreakFlag.PageBreak_AlwaysAfter
55
+ QtPageBreakAuto = QTextFormat.PageBreakFlag.PageBreak_Auto
55
56
 
56
57
  QtPropLineHeight = 1 # QTextBlockFormat.LineHeightTypes.ProportionalHeight
57
58
 
@@ -1,9 +1,9 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: novelWriter
3
- Version: 2.7.1
3
+ Version: 2.7.3
4
4
  Summary: A plain text editor for planning and writing novels
5
5
  Author-email: Veronica Berglyd Olsen <code@vkbo.net>
6
- License: GNU General Public License v3
6
+ License-Expression: GPL-3.0-or-later AND Apache-2.0 AND CC-BY-4.0
7
7
  Project-URL: Homepage, https://novelwriter.io
8
8
  Project-URL: Documentation, https://docs.novelwriter.io
9
9
  Project-URL: Repository, https://github.com/vkbo/novelWriter
@@ -14,7 +14,6 @@ Classifier: Programming Language :: Python :: 3.11
14
14
  Classifier: Programming Language :: Python :: 3.12
15
15
  Classifier: Programming Language :: Python :: 3.13
16
16
  Classifier: Programming Language :: Python :: Implementation :: CPython
17
- Classifier: License :: OSI Approved :: GNU General Public License v3 (GPLv3)
18
17
  Classifier: Development Status :: 5 - Production/Stable
19
18
  Classifier: Operating System :: OS Independent
20
19
  Classifier: Intended Audience :: End Users/Desktop
@@ -23,6 +22,7 @@ Classifier: Topic :: Text Editors
23
22
  Requires-Python: >=3.10
24
23
  Description-Content-Type: text/markdown
25
24
  License-File: LICENSE.md
25
+ License-File: setup/LICENSE-Apache-2.0.txt
26
26
  Requires-Dist: pyqt6>=6.4
27
27
  Requires-Dist: pyenchant>=3.0.0
28
28
  Dynamic: license-file
@@ -1,16 +1,16 @@
1
- novelwriter/__init__.py,sha256=lmuWrq3jOV3xyQVDMcdyl0KpIKi_wR-DO1wTl2lVoWQ,9566
2
- novelwriter/common.py,sha256=Z2n4CNZMotBYNNOqGPzXFfyAALh5eHtV3VtSyJaeBIU,22135
3
- novelwriter/config.py,sha256=frdBJFPhR3tCUkc2dPmk930En0rANy1Om94IWHJtj-M,44665
4
- novelwriter/constants.py,sha256=6gR92ZBcK9J0PUWYaTLlQOaLDo-4Sx3mkbjuiiRT21U,28841
1
+ novelwriter/__init__.py,sha256=-3eKyIaKPxaDAwWINFJyZs8K12e1d7cZEPJqtPy6lUQ,9566
2
+ novelwriter/common.py,sha256=J4WNfcXWdeImTPhkY1vuWoVEQWTpIUE6t7Hwd3GiVzk,22243
3
+ novelwriter/config.py,sha256=PXzcruFnFuqS4zH9RJ0hf0oUs9Vndo-i6aPOBzadsp0,44643
4
+ novelwriter/constants.py,sha256=j_Z3cb-_wW7_cxzTdoRVR9JvmYbioLAp8Dh9TluFJSQ,28827
5
5
  novelwriter/enum.py,sha256=_T1Un3H1SmpfHUw1CEPZuXSKJNT6RQWeN6N5p8yBx8E,3724
6
6
  novelwriter/error.py,sha256=HPKDJb0NPTyVRsrPNG_I1-XfLXlvYAJuE2Xi7maWvXM,6828
7
7
  novelwriter/guimain.py,sha256=luIw3434s92wEuU4gKkerp0MQGtD_hDlCVY6ioZu0LQ,50293
8
8
  novelwriter/shared.py,sha256=8JTkFi7QXmQddGNVaSBIvKPfDXLaJ9XWNmUJblpfXvs,17795
9
9
  novelwriter/splash.py,sha256=rK71vO7NG2NETTf1vvQiWRHfkHonf2Z0xhLj_LF_wj8,2237
10
- novelwriter/types.py,sha256=wehbzAD13D2Ky4fqH3z7jK82uS4dNr19mH94AuLoIOA,5111
11
- novelwriter/assets/manual.pdf,sha256=BC_HyDpgwxdJmf3NLiH_idMr_2HQt3_6TcjhEc8Akgg,2126838
12
- novelwriter/assets/manual_fr.pdf,sha256=cBD_5Z9TWVgq_bpF8iZ2yW_da4h1CYzKMhdwSJ-1ByE,2184930
13
- novelwriter/assets/sample.zip,sha256=zYJMNdjw3d876Sk4nrODAXKbkVUfUR-NMV7nvHYH824,13977
10
+ novelwriter/types.py,sha256=OKugWeCJfxo2JHE_HGA-742nav-ucSPYRI33c9NJqsg,5170
11
+ novelwriter/assets/manual.pdf,sha256=XWZVP5-lBRCk8SMO6_i-6lYUlOaQzfbkYKpQH1ABYI0,2130948
12
+ novelwriter/assets/manual_fr.pdf,sha256=5Zfv290yz7HIv86BAo6Rz8fHX4XnziiUxyakKGqgIJA,2189901
13
+ novelwriter/assets/sample.zip,sha256=Q4JceePb8PjXs36Tug-3txtC4PNvpOMjhsDe_7nMdII,13983
14
14
  novelwriter/assets/i18n/nw_cs_CZ.qm,sha256=EHgCgH3OY_NM1zEnLy3wO7kxARpWk7daTALcBtzuEKw,100334
15
15
  novelwriter/assets/i18n/nw_de_DE.qm,sha256=VwyhNBVst94jlAsdsyUbzj4_ucGX6QLp16wve2jlCjg,105152
16
16
  novelwriter/assets/i18n/nw_en_US.qm,sha256=rK_f6bhwlzqLygB9xfQr9mPLDZAPmutYHr3vie-Kf5k,98000
@@ -89,8 +89,8 @@ novelwriter/core/buildsettings.py,sha256=9dqXlV6oVYe2UmqSrx54PFqvRYcjOJmJEIaFmqj
89
89
  novelwriter/core/coretools.py,sha256=u82B7foOTbkoRXDVshL_Htlup6svTQU6eiIWeJsfsEA,21515
90
90
  novelwriter/core/docbuild.py,sha256=D7YF2aGhpDZ1gGkL8mzXxH1SSilfhoZDy9PVCGZf6s0,14173
91
91
  novelwriter/core/document.py,sha256=AA6Fhwi9Q-UtAFYXjMUXSU-cQF2jogScqPJ5sNaRbEI,12139
92
- novelwriter/core/index.py,sha256=HvYuUA6SAUlce0LygsQ9cP6csRk5rEsZLpZ5JAzSZdc,41199
93
- novelwriter/core/indexdata.py,sha256=ywfbKZEzK2cKWXepM2M8YoAr-G2qSF9L_GV4GIvQB5s,13177
92
+ novelwriter/core/index.py,sha256=Gzg1Co_oNFVx_yBqIN--f-yWGxP6OVH67DsME0-A2qU,41319
93
+ novelwriter/core/indexdata.py,sha256=4FX4DJHfTpJPhiOilPpKVAuj5iDh6OGFI89Z_fQdLE0,13458
94
94
  novelwriter/core/item.py,sha256=4fu4bKIqqCmQamyQnY0LUCO8p0qg371xIQbrVZrxCPU,19677
95
95
  novelwriter/core/itemmodel.py,sha256=wVRUjT_0FGEAzkZFNo4fgCMR4hlQfny2og3gPE3kd8Q,18472
96
96
  novelwriter/core/novelmodel.py,sha256=lAtxerg-BuRydJTqwNggPFoYamTrJc4Z-GVXwq_2zwo,7414
@@ -107,7 +107,7 @@ novelwriter/dialogs/about.py,sha256=hDoXF8kRwwAUBJaZmYqnOggA6zgdZJH8aYLYhWntK7Q,
107
107
  novelwriter/dialogs/docmerge.py,sha256=HaXD5bEfd2AL7JEiXKjSDZklAL1_uaJtbFihMLBlqV8,6082
108
108
  novelwriter/dialogs/docsplit.py,sha256=WLQMB1BdqLEFxEcZ63GSSba7yJb5xBFfaiCdzbC9aK0,9380
109
109
  novelwriter/dialogs/editlabel.py,sha256=SPXp_52HZcYVks_Ap818MKw4nsDme8mmaXi_T1lfJjI,2968
110
- novelwriter/dialogs/preferences.py,sha256=PIKcOFS-hrj97TbimFMdwTk_FMKnXW86DKkudl6DQTA,41923
110
+ novelwriter/dialogs/preferences.py,sha256=qHk8e2q9Q51x8wDo2jfX2_fVoXnxCpxH5EIKDGNW8M8,42910
111
111
  novelwriter/dialogs/projectsettings.py,sha256=gTo9w6-WxTMk533BIPFu4O-3bHpNoVYU1hBhnAC2uJQ,29963
112
112
  novelwriter/dialogs/quotes.py,sha256=X_LIHktkn7r-UgRwv2FalZ3n5_a8FKTW0ceS3iWOmgU,4553
113
113
  novelwriter/dialogs/wordlist.py,sha256=lPZi5fwqOhr12KsvnO4CC1BFtSnjSuTMCbZm5S0JRzk,8507
@@ -122,21 +122,21 @@ novelwriter/extensions/switch.py,sha256=dfsZtJsXvqzkgkCQHRSjNfT6S4P3zy4UmTJ3CKpY
122
122
  novelwriter/extensions/switchbox.py,sha256=QIBgy75OeVwmu53A1sOhlrarSlRH6TxSQNkIyoqFc9Q,4311
123
123
  novelwriter/extensions/versioninfo.py,sha256=8DTvtu4sFPCioY7L-3rK1_sAN1qod3XwXl3F4LtKbDY,5046
124
124
  novelwriter/formats/shared.py,sha256=I_MfvU9ZuN23WblDXdggiV282cXuPDpkq72M3MyK-G8,4666
125
- novelwriter/formats/todocx.py,sha256=WNIW2RJst7ai3C8IKHcQjwbh-KRzoZnhjWwVmjug_yc,40342
125
+ novelwriter/formats/todocx.py,sha256=8N7UlA-c-NCOmiRu0XmyEcxndGbKacjI52I7ZzQ7eXI,40368
126
126
  novelwriter/formats/tohtml.py,sha256=04OjlD_kZJyWBIuUc2olX-jSVh89_lLctST89gWIQMw,17153
127
- novelwriter/formats/tokenizer.py,sha256=UiucWH3OFAyoqNEQMc_IQ5p6egemhMTf5PZrsvyh3lk,48648
127
+ novelwriter/formats/tokenizer.py,sha256=IrkeFgiUYqltTVk1Y5munfoV9BPHpAcOdqpGljc1qGE,48550
128
128
  novelwriter/formats/tomarkdown.py,sha256=CN9mJRwqoit1c9rOHwsqBZEpPjEWYvh81IVgttUxtEU,7044
129
129
  novelwriter/formats/toodt.py,sha256=LuzN8vE46mclcfpOs9QZcElXDVOx14wjyKPkOQE6KYQ,58672
130
- novelwriter/formats/toqdoc.py,sha256=-drPu62rVw_velHTkEEToKiM6-B8PvozX4PPnlDuu_k,18159
130
+ novelwriter/formats/toqdoc.py,sha256=aJGFhq482p7lxr72bCidd2KkpkNH0mcsQvV2AcI1zgo,18315
131
131
  novelwriter/formats/toraw.py,sha256=6qLNvDc87t6c25j1H2IL9whajp1u-SOnHFk9c4gV_7E,2903
132
132
  novelwriter/gui/doceditor.py,sha256=yTi2w289ONN-0YJBUFSPXZXMl2tENo78u0X3HoxCAnA,117541
133
133
  novelwriter/gui/dochighlight.py,sha256=DDCEK9iKUBd5miScrJCj9NlRj3sdOgD_bY8_4ARsSMg,19576
134
- novelwriter/gui/docviewer.py,sha256=5xRmA0Fng7c5Y0hnwHFKzlvLdgoKkg3JxDajylkPW5I,34438
134
+ novelwriter/gui/docviewer.py,sha256=7r24roJ1NcGAXC022Yo3wqdwb3AgpCkX65vN7xxQja8,34569
135
135
  novelwriter/gui/docviewerpanel.py,sha256=m5e9hO5dgV729RnOEJzNzCOU2CB3vPzp4TB4HKgTjGA,19361
136
136
  novelwriter/gui/editordocument.py,sha256=SPWulwJEsG_m_d7cJzAa8fhWmNKLGNPfI-Lsq_mF9CU,4774
137
137
  novelwriter/gui/itemdetails.py,sha256=3mwtSGmOwKkFsFQi6ydmkaPaT4Eecg4fetkG6TduPWk,9605
138
138
  novelwriter/gui/mainmenu.py,sha256=toBoa8tuSKuFrCClGdyI2jVF82kp3V2ZWC3W4xzrfFU,42043
139
- novelwriter/gui/noveltree.py,sha256=-r5SHTKnIpsp-EZ7-JMFnQXzgApLBArsmsKnz-16bg0,20698
139
+ novelwriter/gui/noveltree.py,sha256=HWM1DV6rqRGoJ_5DdQXm9Q99iXWOv5ZuXGPshRWJalA,20758
140
140
  novelwriter/gui/outline.py,sha256=T4hxb_qjn_tl3e_mcBvr5X-PUZjBRYJo4Oga4nQ3FHg,41786
141
141
  novelwriter/gui/projtree.py,sha256=ACteHsSp0Od6s3d9nMxsJwpsCfWCEiewIMZhTujxWH0,53760
142
142
  novelwriter/gui/search.py,sha256=UsqOoDBdgFNxIN2hOSBJppTiJ31OTuoscNpQarNtnDo,13107
@@ -149,14 +149,15 @@ novelwriter/text/patterns.py,sha256=ewlKV9IiKMlLZj4tJk2noSh8V5fzyDK9Cn_vmXo7eEk,
149
149
  novelwriter/tools/dictionaries.py,sha256=ItM080yCoB5lgcovwUU1AO6usDFBRQpcV1UG0KGFvek,8932
150
150
  novelwriter/tools/lipsum.py,sha256=tYSKrrtqCFlpCzXizgiVcG75Zud1hRhoPwtFpKPLL-0,4842
151
151
  novelwriter/tools/manusbuild.py,sha256=FcM36eukyaWTI7G5Yb5bOShoZhoqSHCvJtmmhYng4Eo,13814
152
- novelwriter/tools/manuscript.py,sha256=j9uWJkvMSV8UuSG6gnwflby9GFFaODiqcA7B9EUrxaI,39559
153
- novelwriter/tools/manussettings.py,sha256=Fi1WXCc0LYt4ryR4ae0HAy2Goqzi4WNH0cGxJZ8yPYs,62448
152
+ novelwriter/tools/manuscript.py,sha256=cnTkXgaLlGcXFKJo5UG2fFhKlzfc9zGvEinfzetf91E,39567
153
+ novelwriter/tools/manussettings.py,sha256=pQhugOCzL957M7pqT2szajZzNgnWWA1qFiIHex0RDyQ,62707
154
154
  novelwriter/tools/noveldetails.py,sha256=4BVi3Pc6HNJlYcYnyHKiIuR23uKZ9elNZOcJJOoLlAw,18067
155
155
  novelwriter/tools/welcome.py,sha256=f657EWmQxe3l2sRL3Wq1A9R5mnFciivAF5TDbsUlsRc,27431
156
156
  novelwriter/tools/writingstats.py,sha256=3U4dUlaeNR16E_Kxo5am8ov3ICH0j6S48FLJ-IptHDI,23161
157
- novelwriter-2.7.1.dist-info/licenses/LICENSE.md,sha256=2GirkkLrPfQqx7fACKRJjtKJUegKc8067erGvcDVQHM,32197
158
- novelwriter-2.7.1.dist-info/METADATA,sha256=gPrsEAXN17MdQEGn8DMPRfTGRlTLdzNpBZC6MUBuLHA,2546
159
- novelwriter-2.7.1.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
160
- novelwriter-2.7.1.dist-info/entry_points.txt,sha256=YDUG1w361LtLsjD3YhxoTDTwM17JA5-nigjC6j5C74A,45
161
- novelwriter-2.7.1.dist-info/top_level.txt,sha256=wFFEucjEeNC_Ap5ULBuEutg5a1Uc0-YO9uFT5L2naNI,12
162
- novelwriter-2.7.1.dist-info/RECORD,,
157
+ novelwriter-2.7.3.dist-info/licenses/LICENSE.md,sha256=2GirkkLrPfQqx7fACKRJjtKJUegKc8067erGvcDVQHM,32197
158
+ novelwriter-2.7.3.dist-info/licenses/setup/LICENSE-Apache-2.0.txt,sha256=WNHhf_5RCaeuKWyq_K39vmp9F28LxKsB4SpomwSZ2L0,11357
159
+ novelwriter-2.7.3.dist-info/METADATA,sha256=_nrYCbRF5fxtGVEzcym515fRXd6Ryhlo6WGi5KJGChw,2539
160
+ novelwriter-2.7.3.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
161
+ novelwriter-2.7.3.dist-info/entry_points.txt,sha256=YDUG1w361LtLsjD3YhxoTDTwM17JA5-nigjC6j5C74A,45
162
+ novelwriter-2.7.3.dist-info/top_level.txt,sha256=wFFEucjEeNC_Ap5ULBuEutg5a1Uc0-YO9uFT5L2naNI,12
163
+ novelwriter-2.7.3.dist-info/RECORD,,
@@ -0,0 +1,202 @@
1
+
2
+ Apache License
3
+ Version 2.0, January 2004
4
+ http://www.apache.org/licenses/
5
+
6
+ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
7
+
8
+ 1. Definitions.
9
+
10
+ "License" shall mean the terms and conditions for use, reproduction,
11
+ and distribution as defined by Sections 1 through 9 of this document.
12
+
13
+ "Licensor" shall mean the copyright owner or entity authorized by
14
+ the copyright owner that is granting the License.
15
+
16
+ "Legal Entity" shall mean the union of the acting entity and all
17
+ other entities that control, are controlled by, or are under common
18
+ control with that entity. For the purposes of this definition,
19
+ "control" means (i) the power, direct or indirect, to cause the
20
+ direction or management of such entity, whether by contract or
21
+ otherwise, or (ii) ownership of fifty percent (50%) or more of the
22
+ outstanding shares, or (iii) beneficial ownership of such entity.
23
+
24
+ "You" (or "Your") shall mean an individual or Legal Entity
25
+ exercising permissions granted by this License.
26
+
27
+ "Source" form shall mean the preferred form for making modifications,
28
+ including but not limited to software source code, documentation
29
+ source, and configuration files.
30
+
31
+ "Object" form shall mean any form resulting from mechanical
32
+ transformation or translation of a Source form, including but
33
+ not limited to compiled object code, generated documentation,
34
+ and conversions to other media types.
35
+
36
+ "Work" shall mean the work of authorship, whether in Source or
37
+ Object form, made available under the License, as indicated by a
38
+ copyright notice that is included in or attached to the work
39
+ (an example is provided in the Appendix below).
40
+
41
+ "Derivative Works" shall mean any work, whether in Source or Object
42
+ form, that is based on (or derived from) the Work and for which the
43
+ editorial revisions, annotations, elaborations, or other modifications
44
+ represent, as a whole, an original work of authorship. For the purposes
45
+ of this License, Derivative Works shall not include works that remain
46
+ separable from, or merely link (or bind by name) to the interfaces of,
47
+ the Work and Derivative Works thereof.
48
+
49
+ "Contribution" shall mean any work of authorship, including
50
+ the original version of the Work and any modifications or additions
51
+ to that Work or Derivative Works thereof, that is intentionally
52
+ submitted to Licensor for inclusion in the Work by the copyright owner
53
+ or by an individual or Legal Entity authorized to submit on behalf of
54
+ the copyright owner. For the purposes of this definition, "submitted"
55
+ means any form of electronic, verbal, or written communication sent
56
+ to the Licensor or its representatives, including but not limited to
57
+ communication on electronic mailing lists, source code control systems,
58
+ and issue tracking systems that are managed by, or on behalf of, the
59
+ Licensor for the purpose of discussing and improving the Work, but
60
+ excluding communication that is conspicuously marked or otherwise
61
+ designated in writing by the copyright owner as "Not a Contribution."
62
+
63
+ "Contributor" shall mean Licensor and any individual or Legal Entity
64
+ on behalf of whom a Contribution has been received by Licensor and
65
+ subsequently incorporated within the Work.
66
+
67
+ 2. Grant of Copyright License. Subject to the terms and conditions of
68
+ this License, each Contributor hereby grants to You a perpetual,
69
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
70
+ copyright license to reproduce, prepare Derivative Works of,
71
+ publicly display, publicly perform, sublicense, and distribute the
72
+ Work and such Derivative Works in Source or Object form.
73
+
74
+ 3. Grant of Patent License. Subject to the terms and conditions of
75
+ this License, each Contributor hereby grants to You a perpetual,
76
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
77
+ (except as stated in this section) patent license to make, have made,
78
+ use, offer to sell, sell, import, and otherwise transfer the Work,
79
+ where such license applies only to those patent claims licensable
80
+ by such Contributor that are necessarily infringed by their
81
+ Contribution(s) alone or by combination of their Contribution(s)
82
+ with the Work to which such Contribution(s) was submitted. If You
83
+ institute patent litigation against any entity (including a
84
+ cross-claim or counterclaim in a lawsuit) alleging that the Work
85
+ or a Contribution incorporated within the Work constitutes direct
86
+ or contributory patent infringement, then any patent licenses
87
+ granted to You under this License for that Work shall terminate
88
+ as of the date such litigation is filed.
89
+
90
+ 4. Redistribution. You may reproduce and distribute copies of the
91
+ Work or Derivative Works thereof in any medium, with or without
92
+ modifications, and in Source or Object form, provided that You
93
+ meet the following conditions:
94
+
95
+ (a) You must give any other recipients of the Work or
96
+ Derivative Works a copy of this License; and
97
+
98
+ (b) You must cause any modified files to carry prominent notices
99
+ stating that You changed the files; and
100
+
101
+ (c) You must retain, in the Source form of any Derivative Works
102
+ that You distribute, all copyright, patent, trademark, and
103
+ attribution notices from the Source form of the Work,
104
+ excluding those notices that do not pertain to any part of
105
+ the Derivative Works; and
106
+
107
+ (d) If the Work includes a "NOTICE" text file as part of its
108
+ distribution, then any Derivative Works that You distribute must
109
+ include a readable copy of the attribution notices contained
110
+ within such NOTICE file, excluding those notices that do not
111
+ pertain to any part of the Derivative Works, in at least one
112
+ of the following places: within a NOTICE text file distributed
113
+ as part of the Derivative Works; within the Source form or
114
+ documentation, if provided along with the Derivative Works; or,
115
+ within a display generated by the Derivative Works, if and
116
+ wherever such third-party notices normally appear. The contents
117
+ of the NOTICE file are for informational purposes only and
118
+ do not modify the License. You may add Your own attribution
119
+ notices within Derivative Works that You distribute, alongside
120
+ or as an addendum to the NOTICE text from the Work, provided
121
+ that such additional attribution notices cannot be construed
122
+ as modifying the License.
123
+
124
+ You may add Your own copyright statement to Your modifications and
125
+ may provide additional or different license terms and conditions
126
+ for use, reproduction, or distribution of Your modifications, or
127
+ for any such Derivative Works as a whole, provided Your use,
128
+ reproduction, and distribution of the Work otherwise complies with
129
+ the conditions stated in this License.
130
+
131
+ 5. Submission of Contributions. Unless You explicitly state otherwise,
132
+ any Contribution intentionally submitted for inclusion in the Work
133
+ by You to the Licensor shall be under the terms and conditions of
134
+ this License, without any additional terms or conditions.
135
+ Notwithstanding the above, nothing herein shall supersede or modify
136
+ the terms of any separate license agreement you may have executed
137
+ with Licensor regarding such Contributions.
138
+
139
+ 6. Trademarks. This License does not grant permission to use the trade
140
+ names, trademarks, service marks, or product names of the Licensor,
141
+ except as required for reasonable and customary use in describing the
142
+ origin of the Work and reproducing the content of the NOTICE file.
143
+
144
+ 7. Disclaimer of Warranty. Unless required by applicable law or
145
+ agreed to in writing, Licensor provides the Work (and each
146
+ Contributor provides its Contributions) on an "AS IS" BASIS,
147
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
148
+ implied, including, without limitation, any warranties or conditions
149
+ of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
150
+ PARTICULAR PURPOSE. You are solely responsible for determining the
151
+ appropriateness of using or redistributing the Work and assume any
152
+ risks associated with Your exercise of permissions under this License.
153
+
154
+ 8. Limitation of Liability. In no event and under no legal theory,
155
+ whether in tort (including negligence), contract, or otherwise,
156
+ unless required by applicable law (such as deliberate and grossly
157
+ negligent acts) or agreed to in writing, shall any Contributor be
158
+ liable to You for damages, including any direct, indirect, special,
159
+ incidental, or consequential damages of any character arising as a
160
+ result of this License or out of the use or inability to use the
161
+ Work (including but not limited to damages for loss of goodwill,
162
+ work stoppage, computer failure or malfunction, or any and all
163
+ other commercial damages or losses), even if such Contributor
164
+ has been advised of the possibility of such damages.
165
+
166
+ 9. Accepting Warranty or Additional Liability. While redistributing
167
+ the Work or Derivative Works thereof, You may choose to offer,
168
+ and charge a fee for, acceptance of support, warranty, indemnity,
169
+ or other liability obligations and/or rights consistent with this
170
+ License. However, in accepting such obligations, You may act only
171
+ on Your own behalf and on Your sole responsibility, not on behalf
172
+ of any other Contributor, and only if You agree to indemnify,
173
+ defend, and hold each Contributor harmless for any liability
174
+ incurred by, or claims asserted against, such Contributor by reason
175
+ of your accepting any such warranty or additional liability.
176
+
177
+ END OF TERMS AND CONDITIONS
178
+
179
+ APPENDIX: How to apply the Apache License to your work.
180
+
181
+ To apply the Apache License to your work, attach the following
182
+ boilerplate notice, with the fields enclosed by brackets "[]"
183
+ replaced with your own identifying information. (Don't include
184
+ the brackets!) The text should be enclosed in the appropriate
185
+ comment syntax for the file format. We also recommend that a
186
+ file or class name and description of purpose be included on the
187
+ same "printed page" as the copyright notice for easier
188
+ identification within third-party archives.
189
+
190
+ Copyright [yyyy] [name of copyright owner]
191
+
192
+ Licensed under the Apache License, Version 2.0 (the "License");
193
+ you may not use this file except in compliance with the License.
194
+ You may obtain a copy of the License at
195
+
196
+ http://www.apache.org/licenses/LICENSE-2.0
197
+
198
+ Unless required by applicable law or agreed to in writing, software
199
+ distributed under the License is distributed on an "AS IS" BASIS,
200
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
201
+ See the License for the specific language governing permissions and
202
+ limitations under the License.