novelWriter 2.7rc1__py3-none-any.whl → 2.7.2__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.
Files changed (50) hide show
  1. novelwriter/__init__.py +3 -3
  2. novelwriter/assets/i18n/nw_cs_CZ.qm +0 -0
  3. novelwriter/assets/i18n/nw_de_DE.qm +0 -0
  4. novelwriter/assets/i18n/nw_en_US.qm +0 -0
  5. novelwriter/assets/i18n/nw_es_419.qm +0 -0
  6. novelwriter/assets/i18n/nw_fr_FR.qm +0 -0
  7. novelwriter/assets/i18n/nw_it_IT.qm +0 -0
  8. novelwriter/assets/i18n/nw_ja_JP.qm +0 -0
  9. novelwriter/assets/i18n/nw_nb_NO.qm +0 -0
  10. novelwriter/assets/i18n/nw_nl_NL.qm +0 -0
  11. novelwriter/assets/i18n/nw_pl_PL.qm +0 -0
  12. novelwriter/assets/i18n/nw_pt_BR.qm +0 -0
  13. novelwriter/assets/i18n/nw_ru_RU.qm +0 -0
  14. novelwriter/assets/i18n/nw_zh_CN.qm +0 -0
  15. novelwriter/assets/i18n/project_cs_CZ.json +2 -0
  16. novelwriter/assets/i18n/project_de_DE.json +3 -1
  17. novelwriter/assets/i18n/project_en_US.json +2 -0
  18. novelwriter/assets/i18n/project_it_IT.json +2 -0
  19. novelwriter/assets/i18n/project_ja_JP.json +2 -0
  20. novelwriter/assets/i18n/project_nb_NO.json +2 -0
  21. novelwriter/assets/i18n/project_nn_NO.json +5 -0
  22. novelwriter/assets/i18n/project_pl_PL.json +2 -0
  23. novelwriter/assets/i18n/project_pt_BR.json +2 -0
  24. novelwriter/assets/i18n/project_ru_RU.json +2 -0
  25. novelwriter/assets/i18n/project_zh_CN.json +2 -0
  26. novelwriter/assets/manual.pdf +0 -0
  27. novelwriter/assets/manual_fr.pdf +0 -0
  28. novelwriter/assets/sample.zip +0 -0
  29. novelwriter/assets/text/credits_en.htm +6 -0
  30. novelwriter/config.py +2 -2
  31. novelwriter/constants.py +16 -19
  32. novelwriter/core/buildsettings.py +1 -1
  33. novelwriter/core/index.py +24 -16
  34. novelwriter/core/indexdata.py +9 -3
  35. novelwriter/core/item.py +1 -1
  36. novelwriter/formats/todocx.py +2 -2
  37. novelwriter/formats/tokenizer.py +45 -39
  38. novelwriter/formats/toqdoc.py +10 -8
  39. novelwriter/gui/doceditor.py +8 -7
  40. novelwriter/gui/docviewer.py +2 -0
  41. novelwriter/gui/noveltree.py +1 -0
  42. novelwriter/tools/manuscript.py +4 -7
  43. novelwriter/tools/manussettings.py +33 -22
  44. novelwriter/types.py +1 -0
  45. {novelwriter-2.7rc1.dist-info → novelwriter-2.7.2.dist-info}/METADATA +3 -4
  46. {novelwriter-2.7rc1.dist-info → novelwriter-2.7.2.dist-info}/RECORD +50 -50
  47. {novelwriter-2.7rc1.dist-info → novelwriter-2.7.2.dist-info}/WHEEL +1 -1
  48. {novelwriter-2.7rc1.dist-info → novelwriter-2.7.2.dist-info}/entry_points.txt +0 -0
  49. {novelwriter-2.7rc1.dist-info → novelwriter-2.7.2.dist-info}/licenses/LICENSE.md +0 -0
  50. {novelwriter-2.7rc1.dist-info → novelwriter-2.7.2.dist-info}/top_level.txt +0 -0
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.7rc1"
53
- __hexversion__ = "0x020700c1"
54
- __date__ = "2025-05-19"
52
+ __version__ = "2.7.2"
53
+ __hexversion__ = "0x020702f0"
54
+ __date__ = "2025-06-24"
55
55
  __status__ = "Stable"
56
56
  __domain__ = "novelwriter.io"
57
57
 
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
@@ -3,6 +3,8 @@
3
3
  "Short Description": "Krátký popis",
4
4
  "Footnotes": "Poznámky pod čarou",
5
5
  "Comment": "Komentář",
6
+ "Story Structure": "Struktura příběhu",
7
+ "Note": "Poznámka",
6
8
  "Notes": "Poznámka",
7
9
  "Tag": "Štítek",
8
10
  "Point of View": "Úhel pohledu",
@@ -1,8 +1,10 @@
1
1
  {
2
- "Synopsis": "Synopsis",
2
+ "Synopsis": "Zusammenfassung",
3
3
  "Short Description": "Kurzbeschreibung",
4
4
  "Footnotes": "Fußnoten",
5
5
  "Comment": "Kommentar",
6
+ "Story Structure": "Struktur",
7
+ "Note": "Notiz",
6
8
  "Notes": "Notizen",
7
9
  "Tag": "Schlagwort",
8
10
  "Point of View": "Perspektive",
@@ -3,6 +3,8 @@
3
3
  "Short Description": "Short Description",
4
4
  "Footnotes": "Footnotes",
5
5
  "Comment": "Comment",
6
+ "Story Structure": "Story Structure",
7
+ "Note": "Note",
6
8
  "Notes": "Notes",
7
9
  "Tag": "Tag",
8
10
  "Point of View": "Point of View",
@@ -3,6 +3,8 @@
3
3
  "Short Description": "Breve descrizione",
4
4
  "Footnotes": "Note a piè pagina",
5
5
  "Comment": "Commento",
6
+ "Story Structure": "Struttura della storia",
7
+ "Note": "Nota",
6
8
  "Notes": "Note",
7
9
  "Tag": "Etichetta",
8
10
  "Point of View": "Punto di vista",
@@ -3,6 +3,8 @@
3
3
  "Short Description": "短い説明",
4
4
  "Footnotes": "脚注",
5
5
  "Comment": "コメント ",
6
+ "Story Structure": "ストーリー構造",
7
+ "Note": "ノート",
6
8
  "Notes": "ノート",
7
9
  "Tag": "タグ",
8
10
  "Point of View": "視点",
@@ -3,6 +3,8 @@
3
3
  "Short Description": "Kort beskrivelse",
4
4
  "Footnotes": "Fotnoter",
5
5
  "Comment": "Kommentar",
6
+ "Story Structure": "Tekst-element",
7
+ "Note": "Notat",
6
8
  "Notes": "Notat",
7
9
  "Tag": "Knagg",
8
10
  "Point of View": "Perspektiv",
@@ -1,6 +1,10 @@
1
1
  {
2
2
  "Synopsis": "Samandrag",
3
+ "Short Description": "Kort beskriving",
4
+ "Footnotes": "Fotnotar",
3
5
  "Comment": "Kommentar",
6
+ "Story Structure": "Tekst-element",
7
+ "Note": "Notat",
4
8
  "Notes": "Notat",
5
9
  "Tag": "Knagg",
6
10
  "Point of View": "Perspektiv",
@@ -12,6 +16,7 @@
12
16
  "Objects": "Objekt",
13
17
  "Entities": "Eining",
14
18
  "Custom": "Anna",
19
+ "New Page": "Ny side",
15
20
  "0": "null",
16
21
  "1": "ein",
17
22
  "2": "to",
@@ -3,6 +3,8 @@
3
3
  "Short Description": "Krótki opis",
4
4
  "Footnotes": "Przypisy",
5
5
  "Comment": "Komentarz",
6
+ "Story Structure": "Struktura opowieści",
7
+ "Note": "Notatka",
6
8
  "Notes": "Notatki",
7
9
  "Tag": "Znacznik",
8
10
  "Point of View": "Punkt widzenia",
@@ -3,6 +3,8 @@
3
3
  "Short Description": "Descrição breve",
4
4
  "Footnotes": "Notas de rodapé",
5
5
  "Comment": "Comentário",
6
+ "Story Structure": "Estrutura narrativa",
7
+ "Note": "Nota",
6
8
  "Notes": "Notas",
7
9
  "Tag": "Etiqueta",
8
10
  "Point of View": "Ponto de vista",
@@ -1,6 +1,7 @@
1
1
  {
2
2
  "Synopsis": "Сводка",
3
3
  "Short Description": "Краткое описание",
4
+ "Footnotes": "Примечания",
4
5
  "Comment": "Комментарий",
5
6
  "Notes": "Заметки",
6
7
  "Tag": "Тэг",
@@ -13,6 +14,7 @@
13
14
  "Objects": "Объекты",
14
15
  "Entities": "Сущности",
15
16
  "Custom": "Другое",
17
+ "New Page": "Новая страница",
16
18
  "0": "Ноль",
17
19
  "1": "Один",
18
20
  "2": "Два",
@@ -3,6 +3,8 @@
3
3
  "Short Description": "简短描述",
4
4
  "Footnotes": "注释",
5
5
  "Comment": "注释",
6
+ "Story Structure": "故事结构",
7
+ "Note": "笔记",
6
8
  "Notes": "笔记",
7
9
  "Tag": "标签",
8
10
  "Point of View": "视角",
Binary file
Binary file
Binary file
@@ -47,6 +47,12 @@ translators for the languages currently available:</p>
47
47
  <li><b>Portuguese:</b> Oli Maia (olimaia)</li>
48
48
  </ul>
49
49
 
50
+ <p>Translations of the documentation:</p>
51
+
52
+ <ul>
53
+ <li><b>French:</b> Jean-Michel Heras (Karduin)</li>
54
+ </ul>
55
+
50
56
  <p>Translations are managed on <a href="https://crowdin.com/project/novelwriter">Crowdin</a>, and
51
57
  more contributions are listed on the project's Members page.</p>
52
58
 
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
@@ -184,6 +184,10 @@ class nwKeyWords:
184
184
  POV_KEY, FOCUS_KEY, CHAR_KEY, PLOT_KEY, TIME_KEY, WORLD_KEY,
185
185
  OBJECT_KEY, ENTITY_KEY, CUSTOM_KEY,
186
186
  ]
187
+ CAN_LOOKUP: Final[list[str]] = [
188
+ POV_KEY, FOCUS_KEY, CHAR_KEY, PLOT_KEY, TIME_KEY, WORLD_KEY,
189
+ OBJECT_KEY, ENTITY_KEY, CUSTOM_KEY, STORY_KEY, MENTION_KEY,
190
+ ]
187
191
 
188
192
  # Set of Valid Keys
189
193
  VALID_KEYS: Final[set[str]] = set(ALL_KEYS)
@@ -582,21 +586,14 @@ class nwUnicode:
582
586
  U_TIMES = "\u00d7" # Multiplication sign
583
587
  U_DIVIDE = "\u00f7" # Division sign
584
588
 
585
- # Arrows
586
- U_UTRI = "\u25b2" # Up-pointing triangle
587
- U_UTRIS = "\u25b4" # Up-pointing triangle, small
588
- U_RTRI = "\u25b6" # Right-pointing triangle
589
- U_RTRIS = "\u25b8" # Right-pointing triangle, small
590
- U_DTRI = "\u25bc" # Down-pointing triangle
591
- U_DTRIS = "\u25be" # Down-pointing triangle, small
592
- U_LTRI = "\u25c0" # Left-pointing triangle
593
- U_LTRIS = "\u25c2" # Left-pointing triangle, small
594
-
595
589
  # Special
596
590
  U_UNKN = "\ufffd" # Unknown character
597
591
  U_NAC1 = "\ufffe" # Not a character
598
592
  U_NAC2 = "\uffff" # Not a character
599
593
 
594
+ # Placeholders
595
+ U_LBREAK = "\u21b2" # Downwards Arrow With Tip Leftwards
596
+
600
597
  # HTML Equivalents
601
598
  # ================
602
599
 
@@ -651,15 +648,15 @@ class nwUnicode:
651
648
  H_TIMES = "&times;"
652
649
  H_DIVIDE = "&divide;"
653
650
 
654
- # Arrows
655
- H_UTRI = "&#9650;"
656
- H_UTRIS = "&#9652;"
657
- H_RTRI = "&#9654;"
658
- H_RTRIS = "&#9656;"
659
- H_DTRI = "&#9660;"
660
- H_DTRIS = "&#9662;"
661
- H_LTRI = "&#9664;"
662
- H_LTRIS = "&#9666;"
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
+ ]
663
660
 
664
661
 
665
662
  class nwHtmlUnicode:
@@ -390,7 +390,7 @@ class BuildSettings:
390
390
  def setValue(self, key: str, value: T_BuildValue) -> None:
391
391
  """Set a specific value for a build setting."""
392
392
  if (d := SETTINGS_TEMPLATE.get(key)) and len(d) == 2 and isinstance(value, d[0]):
393
- self._changed = value != self._settings[key]
393
+ self._changed |= (value != self._settings[key])
394
394
  self._settings[key] = value
395
395
  return
396
396
 
novelwriter/core/index.py CHANGED
@@ -131,12 +131,6 @@ class Index:
131
131
  self._novelExtra = extra
132
132
  return
133
133
 
134
- def setItemClass(self, tHandle: str, itemClass: nwItemClass) -> None:
135
- """Update the class for all tags of a handle."""
136
- logger.info("Updating class for '%s'", tHandle)
137
- self._tagsIndex.updateClass(tHandle, itemClass.name)
138
- return
139
-
140
134
  ##
141
135
  # Public Methods
142
136
  ##
@@ -183,6 +177,16 @@ class Index:
183
177
  self.scanText(tHandle, self._project.storage.getDocumentText(tHandle))
184
178
  return
185
179
 
180
+ def refreshHandle(self, tHandle: str) -> None:
181
+ """Update the class for all tags of a handle."""
182
+ if item := self._project.tree[tHandle]:
183
+ logger.info("Updating class for '%s'", tHandle)
184
+ if item.isInactiveClass():
185
+ self.deleteHandle(tHandle)
186
+ else:
187
+ self._tagsIndex.updateClass(tHandle, item.itemClass.name)
188
+ return
189
+
186
190
  def indexChangedSince(self, checkTime: int | float) -> bool:
187
191
  """Check if the index has changed since a given time."""
188
192
  return self._indexChange > float(checkTime)
@@ -704,17 +708,19 @@ class Index:
704
708
  return 0, 0, 0
705
709
 
706
710
  def getReferences(self, tHandle: str, sTitle: str | None = None) -> dict[str, list[str]]:
707
- """Extract all references made in a file, and optionally title
708
- section.
711
+ """Extract all tags and references made in a file, and
712
+ optionally title section.
709
713
  """
710
- tRefs = {x: [] for x in nwKeyWords.VALID_KEYS}
714
+ refs = {x: [] for x in nwKeyWords.VALID_KEYS}
711
715
  for rTitle, hItem in self._itemIndex.iterItemHeaders(tHandle):
712
716
  if sTitle is None or sTitle == rTitle:
713
717
  for aTag, refTypes in hItem.references.items():
714
718
  for refType in refTypes:
715
- if refType in tRefs:
716
- tRefs[refType].append(self._tagsIndex.tagName(aTag))
717
- 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
718
724
 
719
725
  def getReferenceForHeader(self, tHandle: str, nHead: int, keyClass: str) -> list[str]:
720
726
  """Get the display names for a tags class for insertion into a
@@ -753,10 +759,12 @@ class Index:
753
759
  """Return all tags used by a specific document."""
754
760
  return self._itemIndex.allItemTags(tHandle) if tHandle else []
755
761
 
756
- def getClassTags(self, itemClass: nwItemClass | None) -> list[str]:
757
- """Return all tags based on itemClass."""
758
- name = None if itemClass is None else itemClass.name
759
- return self._tagsIndex.filterTagNames(name)
762
+ def getKeyWordTags(self, keyWord: str) -> list[str]:
763
+ """Return all tags usable for a specific keyword."""
764
+ if keyWord in nwKeyWords.CAN_LOOKUP:
765
+ itemClass = nwKeyWords.KEY_CLASS.get(keyWord)
766
+ return self._tagsIndex.filterTagNames(itemClass.name if itemClass else None)
767
+ return []
760
768
 
761
769
  def getTagsData(
762
770
  self, activeOnly: bool = True
@@ -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
  ##
novelwriter/core/item.py CHANGED
@@ -439,7 +439,7 @@ class NWItem:
439
439
  self.setClass(itemClass)
440
440
  if self._type == nwItemType.FILE:
441
441
  # Notify the index of the class change
442
- self._project.index.setItemClass(self._handle, itemClass)
442
+ self._project.index.refreshHandle(self._handle)
443
443
 
444
444
  if self._layout == nwItemLayout.NO_LAYOUT:
445
445
  # If no layout is set, pick one
@@ -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
 
@@ -1090,11 +1090,14 @@ class GuiDocEditor(QPlainTextEdit):
1090
1090
  show = self._completer.updateMetaText(text, bPos)
1091
1091
  else:
1092
1092
  show = self._completer.updateCommentText(text, bPos)
1093
- point = self.cursorRect().bottomRight()
1094
- self._completer.move(viewport.mapToGlobal(point))
1095
- self._completer.setVisible(show)
1093
+ if show:
1094
+ point = self.cursorRect().bottomRight()
1095
+ self._completer.move(viewport.mapToGlobal(point))
1096
+ self._completer.show()
1097
+ else:
1098
+ self._completer.close()
1096
1099
  else:
1097
- self._completer.setVisible(False)
1100
+ self._completer.close()
1098
1101
 
1099
1102
  if self._doReplace and added == 1:
1100
1103
  cursor = self.textCursor()
@@ -2113,9 +2116,7 @@ class CommandCompleter(QMenu):
2113
2116
  length = len(lookup)
2114
2117
  suffix = ""
2115
2118
  options = sorted(filter(
2116
- lambda x: lookup in x.lower(), SHARED.project.index.getClassTags(
2117
- nwKeyWords.KEY_CLASS.get(kw.strip())
2118
- )
2119
+ lambda x: lookup in x.lower(), SHARED.project.index.getKeyWordTags(kw.strip())
2119
2120
  ))[:15]
2120
2121
 
2121
2122
  if not options:
@@ -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
@@ -80,7 +80,8 @@ class GuiBuildSettings(NToolDialog):
80
80
  logger.debug("Create: GuiBuildSettings")
81
81
  self.setObjectName("GuiBuildSettings")
82
82
 
83
- self._build = build
83
+ # Make a copy of the build object
84
+ self._build = BuildSettings.fromDict(build.pack())
84
85
 
85
86
  self.setWindowTitle(self.tr("Manuscript Build Settings"))
86
87
  self.setMinimumSize(700, 400)
@@ -184,6 +185,7 @@ class GuiBuildSettings(NToolDialog):
184
185
  settings.
185
186
  """
186
187
  logger.debug("Closing: GuiBuildSettings")
188
+ self._applyChanges()
187
189
  self._askToSaveBuild()
188
190
  self._saveSettings()
189
191
  event.accept()
@@ -211,11 +213,14 @@ class GuiBuildSettings(NToolDialog):
211
213
  """Handle button clicks from the dialog button box."""
212
214
  role = self.buttonBox.buttonRole(button)
213
215
  if role == QtRoleApply:
216
+ self._applyChanges()
214
217
  self._emitBuildData()
215
218
  elif role == QtRoleAccept:
219
+ self._applyChanges()
216
220
  self._emitBuildData()
217
221
  self.close()
218
222
  elif role == QtRoleReject:
223
+ self._build.resetChangedState()
219
224
  self.close()
220
225
  return
221
226
 
@@ -228,10 +233,9 @@ class GuiBuildSettings(NToolDialog):
228
233
  whether the user wants to save them.
229
234
  """
230
235
  if self._build.changed:
231
- response = SHARED.question(self.tr(
236
+ if SHARED.question(self.tr(
232
237
  "Do you want to save your changes to '{0}'?"
233
- ).format(self._build.name))
234
- if response:
238
+ ).format(self._build.name)):
235
239
  self._emitBuildData()
236
240
  self._build.resetChangedState()
237
241
  return
@@ -246,14 +250,17 @@ class GuiBuildSettings(NToolDialog):
246
250
  pOptions.setValue("GuiBuildSettings", "treeWidth", treeWidth)
247
251
  pOptions.setValue("GuiBuildSettings", "filterWidth", filterWidth)
248
252
  pOptions.saveSettings()
249
-
250
253
  return
251
254
 
252
- def _emitBuildData(self) -> None:
253
- """Assemble the build data and emit the signal."""
255
+ def _applyChanges(self) -> None:
256
+ """Apply all settings changes to the build object."""
254
257
  self._build.setName(self.editBuildName.text())
255
258
  self.optTabHeadings.saveContent()
256
259
  self.optTabFormatting.saveContent()
260
+ return
261
+
262
+ def _emitBuildData(self) -> None:
263
+ """Assemble the build data and emit the signal."""
257
264
  self.newSettingsReady.emit(self._build)
258
265
  self._build.resetChangedState()
259
266
  return
@@ -788,12 +795,15 @@ class _HeadingsTab(NScrollablePage):
788
795
 
789
796
  def loadContent(self) -> None:
790
797
  """Populate the widgets."""
791
- self.fmtPart.setText(self._build.getStr("headings.fmtPart"))
792
- self.fmtChapter.setText(self._build.getStr("headings.fmtChapter"))
793
- self.fmtUnnumbered.setText(self._build.getStr("headings.fmtUnnumbered"))
794
- self.fmtScene.setText(self._build.getStr("headings.fmtScene"))
795
- self.fmtAScene.setText(self._build.getStr("headings.fmtAltScene"))
796
- 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")))
797
807
 
798
808
  self.swtPart.setChecked(self._build.getBool("headings.hidePart"))
799
809
  self.swtChapter.setChecked(self._build.getBool("headings.hideChapter"))
@@ -873,7 +883,7 @@ class _HeadingsTab(NScrollablePage):
873
883
  text = ""
874
884
  label = self.tr("None")
875
885
 
876
- self.editTextBox.setPlainText(text.replace(nwHeadFmt.BR, "\n"))
886
+ self.editTextBox.setPlainText(text.replace(nwUnicode.U_LBREAK, "\n"))
877
887
  self.lblEditForm.setText(self.tr("Editing: {0}").format(label))
878
888
 
879
889
  return
@@ -886,25 +896,26 @@ class _HeadingsTab(NScrollablePage):
886
896
  def _saveFormat(self) -> None:
887
897
  """Save the format from the edit text box."""
888
898
  heading = self._editing
889
- 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)
890
901
  if heading == self.EDIT_TITLE:
891
902
  self.fmtPart.setText(text)
892
- self._build.setValue("headings.fmtPart", text)
903
+ self._build.setValue("headings.fmtPart", value)
893
904
  elif heading == self.EDIT_CHAPTER:
894
905
  self.fmtChapter.setText(text)
895
- self._build.setValue("headings.fmtChapter", text)
906
+ self._build.setValue("headings.fmtChapter", value)
896
907
  elif heading == self.EDIT_UNNUM:
897
908
  self.fmtUnnumbered.setText(text)
898
- self._build.setValue("headings.fmtUnnumbered", text)
909
+ self._build.setValue("headings.fmtUnnumbered", value)
899
910
  elif heading == self.EDIT_SCENE:
900
911
  self.fmtScene.setText(text)
901
- self._build.setValue("headings.fmtScene", text)
912
+ self._build.setValue("headings.fmtScene", value)
902
913
  elif heading == self.EDIT_HSCENE:
903
914
  self.fmtAScene.setText(text)
904
- self._build.setValue("headings.fmtAltScene", text)
915
+ self._build.setValue("headings.fmtAltScene", value)
905
916
  elif heading == self.EDIT_SECTION:
906
917
  self.fmtSection.setText(text)
907
- self._build.setValue("headings.fmtSection", text)
918
+ self._build.setValue("headings.fmtSection", value)
908
919
  else:
909
920
  return
910
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.7rc1
4
- Summary: A markdown-like text editor for planning and writing novels
3
+ Version: 2.7.2
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
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
@@ -1,44 +1,44 @@
1
- novelwriter/__init__.py,sha256=iWZXwrT1Rsgvh5CW08kt_PNdtUZFu0JRU4o6gX0SFZ4,9567
1
+ novelwriter/__init__.py,sha256=o6wvJgoXi9r4JQio7x0dg0lOAy1teJSjQB9DOz6PGkI,9566
2
2
  novelwriter/common.py,sha256=Z2n4CNZMotBYNNOqGPzXFfyAALh5eHtV3VtSyJaeBIU,22135
3
- novelwriter/config.py,sha256=frdBJFPhR3tCUkc2dPmk930En0rANy1Om94IWHJtj-M,44665
4
- novelwriter/constants.py,sha256=yDXImT9FTlMWi-sHCAp-UqILus4Ppg7o776ooudWGG0,28661
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=7VY7ZjdQlnlQWMXMZ7KgFOnCBWYJYYYICjdOhunGY30,2746841
12
- novelwriter/assets/manual_fr.pdf,sha256=jofwAfwbxOty1DYByCqfwQfwdK9EOBC6qMfwSOHNMfg,2789961
13
- novelwriter/assets/sample.zip,sha256=Q04XSfHUdpH_OEkUIPXg4_Ckkkxb5Q74Ab8rguEgOBY,13970
14
- novelwriter/assets/i18n/nw_cs_CZ.qm,sha256=wqS376aERqXduw3lc7FFhcWapwcn9NYdTICkZYoDI1w,97499
15
- novelwriter/assets/i18n/nw_de_DE.qm,sha256=xKyyeNEd02EquAF7KSiHr4GUccfagD74O_gipoQK1Io,102467
16
- novelwriter/assets/i18n/nw_en_US.qm,sha256=fRG9Tg0a9IwsinP1wU162P_Y7mNl-grTKKx3TzKndSI,95213
17
- novelwriter/assets/i18n/nw_es_419.qm,sha256=enE-iIGb_7DypKUXj9o4vddauaR6NO5KO9aAoh9GsEI,105334
18
- novelwriter/assets/i18n/nw_fr_FR.qm,sha256=nIuseguPyp-BW-2QaxZTgTEAq82epUZyAsCpm3Vl0Wg,105307
19
- novelwriter/assets/i18n/nw_it_IT.qm,sha256=eQKNQVvvwO_ja95sSXvZMkAv2TCxBIDYsMcT3k3KeC4,105449
20
- novelwriter/assets/i18n/nw_ja_JP.qm,sha256=fMTV1RI_GhpStiX8RlkCIsyawz6tfcSfrFzuHzS-zpU,78818
21
- novelwriter/assets/i18n/nw_nb_NO.qm,sha256=HIwt18aTorGzdk0j98XJtr652Hd3mCPZi8Ee771-vZw,97781
22
- novelwriter/assets/i18n/nw_nl_NL.qm,sha256=igF8xjFTb4BBnyFThnLCqmk_KQLP3KcDWdUvjfwdBD8,96665
23
- novelwriter/assets/i18n/nw_pl_PL.qm,sha256=etp4w60penSQTTsUavoWFcz1G8vtrV4RPAHTWloCrEg,100719
24
- novelwriter/assets/i18n/nw_pt_BR.qm,sha256=LKeE4hvMlGRk0VIhuIuOkWcOy0SRf0fgQRbhuLQ0dgk,103757
25
- novelwriter/assets/i18n/nw_ru_RU.qm,sha256=T44JdPKRuI27Az7ZjVznT-O4pR941-iVlV6agR-ydcI,94800
26
- novelwriter/assets/i18n/nw_zh_CN.qm,sha256=_MbLHUgM3jBnDBU_fsB3vQlAw8YeWQKpidgJ5p-jiIo,70468
27
- novelwriter/assets/i18n/project_cs_CZ.json,sha256=0A71su1wmPrGD1DD3FruFqnoWQqo4XaggWp5NJKT9lc,2861
28
- novelwriter/assets/i18n/project_de_DE.json,sha256=-9-FF2zYD9ulez9dXAKLbFWsbEX2KLXKE1cJS8csx1g,2866
10
+ novelwriter/types.py,sha256=OKugWeCJfxo2JHE_HGA-742nav-ucSPYRI33c9NJqsg,5170
11
+ novelwriter/assets/manual.pdf,sha256=6Mk9l9LH3cXyIa2j2n5ijD8Z8rV9_dLU4TDJuKl6zWs,2130791
12
+ novelwriter/assets/manual_fr.pdf,sha256=loUjnjQBX8ivzoq9bhmQoY7vkG9ozosJ-on7duvByqU,2189719
13
+ novelwriter/assets/sample.zip,sha256=NxXcUM6EHjhhix4No9-7GTl64_JEGXph_KQmZYmKQak,13987
14
+ novelwriter/assets/i18n/nw_cs_CZ.qm,sha256=EHgCgH3OY_NM1zEnLy3wO7kxARpWk7daTALcBtzuEKw,100334
15
+ novelwriter/assets/i18n/nw_de_DE.qm,sha256=VwyhNBVst94jlAsdsyUbzj4_ucGX6QLp16wve2jlCjg,105152
16
+ novelwriter/assets/i18n/nw_en_US.qm,sha256=rK_f6bhwlzqLygB9xfQr9mPLDZAPmutYHr3vie-Kf5k,98000
17
+ novelwriter/assets/i18n/nw_es_419.qm,sha256=ETykfMUyedjTaRBu-kX6PgGRV79fxjB8cHOzs0rBFeo,104187
18
+ novelwriter/assets/i18n/nw_fr_FR.qm,sha256=7AZLfs_H5gcvqd_gxDE9HjhjVj5Z5TRc-Wsd0lqCFUI,104606
19
+ novelwriter/assets/i18n/nw_it_IT.qm,sha256=TyNTA_U81F5tpk5O7gh9h5mNiFabcjRSEn2pwwi2cIk,108494
20
+ novelwriter/assets/i18n/nw_ja_JP.qm,sha256=rVNa0PlV82Io2MNsYNSf7zta_GfkDK-6gewxr9IFb68,81033
21
+ novelwriter/assets/i18n/nw_nb_NO.qm,sha256=r59Uxw2Tf5n3ZEiBjft4GOCVTgP6_4uusxkmbTBw8-4,100556
22
+ novelwriter/assets/i18n/nw_nl_NL.qm,sha256=kPI2f137gK8qc0boB_geGzN0-ApHr9z3RabUoVM_SOc,95706
23
+ novelwriter/assets/i18n/nw_pl_PL.qm,sha256=mK1B4Yr-H-WSPNccVwCYlcBsGHGOhaYWtPdm--P5nP0,103590
24
+ novelwriter/assets/i18n/nw_pt_BR.qm,sha256=EeJWctUfqbvPgvO_HJQTEB3LaIhCpTM3RKSnCHsly9I,106744
25
+ novelwriter/assets/i18n/nw_ru_RU.qm,sha256=rNKPtUFdeEsYnCiT8qtDWWwG6BrHy1Q_VVzL6ZH0cbo,93879
26
+ novelwriter/assets/i18n/nw_zh_CN.qm,sha256=lYUTXvHrsbLSYw8iI2uevDe9IL0TDkyOATfgvd5f6yw,73077
27
+ novelwriter/assets/i18n/project_cs_CZ.json,sha256=O6UaWMrmmqq1eE7eACWu4lKZPNtmyeXX5tw99PkPWIY,2929
28
+ novelwriter/assets/i18n/project_de_DE.json,sha256=flm9XAftVXNHddhcJEm1MhpsK7yWmnOmh6H12Byk6GA,2925
29
29
  novelwriter/assets/i18n/project_en_GB.json,sha256=sioYhKDpqyuqgSwoyt4CVu6eUNS3ZB_Pr9z7_ufHoPo,2595
30
- novelwriter/assets/i18n/project_en_US.json,sha256=CvBOO8kaW0k7NlwvxFi62Br9HYKJKQLDk1nQfvXOj48,2537
30
+ novelwriter/assets/i18n/project_en_US.json,sha256=sioYhKDpqyuqgSwoyt4CVu6eUNS3ZB_Pr9z7_ufHoPo,2595
31
31
  novelwriter/assets/i18n/project_es_419.json,sha256=NQa1iAyQaoNyFSAfrPXNaoKyStyw7nexnGUHf-UMkOo,2792
32
32
  novelwriter/assets/i18n/project_fr_FR.json,sha256=6ATiNU0YGEXwx_c2LpDvP7eCUDdUiGxtrssCVTiiyC8,2859
33
- novelwriter/assets/i18n/project_it_IT.json,sha256=LHbGZT6MOHOmIKrl6HD1-Tl_RxKbeEL0eXqPVK0VwM8,2633
34
- novelwriter/assets/i18n/project_ja_JP.json,sha256=CK8yvTAr_58mDMSU-XAXYk3ny9Iryp21EZ80dGJ60E8,2451
35
- novelwriter/assets/i18n/project_nb_NO.json,sha256=CS4nU-8AnnmQLiET5wAsFa4F7UHimjCTbxzmtGugw0A,2350
33
+ novelwriter/assets/i18n/project_it_IT.json,sha256=NYUUVp3o-hmahj4oyif8bxVauUMzL8w-XvuR0oGR9ic,2698
34
+ novelwriter/assets/i18n/project_ja_JP.json,sha256=96kVDrvqBQsv_IzTDnoK-IGYZ9jHNXJ15YYhSwaOyFE,2520
35
+ novelwriter/assets/i18n/project_nb_NO.json,sha256=ZaoUbHtPnM30zPfdwPBxProWkR8EP4QykVA1nfrR4_c,2407
36
36
  novelwriter/assets/i18n/project_nl_NL.json,sha256=wlXDfy4rMKyXP0X3r22qSNFuseO33bePAHqvIVk_tHQ,2742
37
- novelwriter/assets/i18n/project_nn_NO.json,sha256=7ZfmtFWHcDMiLVqdOEDHQWHbpoC_bv5ESmf_-CoYiks,2250
38
- novelwriter/assets/i18n/project_pl_PL.json,sha256=2mAchml-aT95ZT9qi3pAEMyyze5swkRa-OCWq6ZMoGk,3517
39
- novelwriter/assets/i18n/project_pt_BR.json,sha256=lCtbr_i2gN3Atii1EgMn-B5SMP71_x4YsF7wXx7Hi7s,2828
40
- novelwriter/assets/i18n/project_ru_RU.json,sha256=Lq8msxzjhkFHXPJAmEEyjoFOUV28CLJXmayY3h3jiAo,4075
41
- novelwriter/assets/i18n/project_zh_CN.json,sha256=CGnJO6ElV40fm7yElNtDwHYsd_32d-cR7h1jG4xx4eY,2377
37
+ novelwriter/assets/i18n/project_nn_NO.json,sha256=Nn_p7Fb0ASKHAZ3l8I0UJuS4GmG6MFInP68RD4han24,2401
38
+ novelwriter/assets/i18n/project_pl_PL.json,sha256=-yWlx1zDYRKhMxMIe-59ixqUeIimQTGFAoq8qe4TiaM,3583
39
+ novelwriter/assets/i18n/project_pt_BR.json,sha256=jZ4PJt2wbLmjNZshAJtiVRlWMQfXHcQTCXxqFVwCTDg,2890
40
+ novelwriter/assets/i18n/project_ru_RU.json,sha256=6Q0H7_cDuobK05N7q8we3VnxXhGqjuG-ulIUESU0lJA,4159
41
+ novelwriter/assets/i18n/project_zh_CN.json,sha256=rvvwclPNI2dMd64kUb7hr7zlkT-kNZ0Bbk7XMHpIcAI,2434
42
42
  novelwriter/assets/icons/font_awesome.icons,sha256=xe54nRX90DbcA2DGTYUEPagbGgrLQlusuJa0TvBmUbg,50737
43
43
  novelwriter/assets/icons/material_filled_bold.icons,sha256=RBlkfnJ46uVq3HZbj7Wdfktc8wSexktao1E_H2puKH8,83050
44
44
  novelwriter/assets/icons/material_filled_normal.icons,sha256=CTsuQKjph62heUboAIqin5LGZogFWbQ7gW--rvZDz_I,52676
@@ -75,7 +75,7 @@ novelwriter/assets/syntax/tomorrow_night.conf,sha256=mWPneIV_6tszLEK4-Eo-kcJmUbr
75
75
  novelwriter/assets/syntax/tomorrow_night_blue.conf,sha256=L42gcekFfNpCCB45m9Lhzmz8Y6Lvbpik3AADi_1Aao8,1549
76
76
  novelwriter/assets/syntax/tomorrow_night_bright.conf,sha256=h8_775Ed8hl2XhfCoxAY2kT43vi-UvG8UHik1aI4eRw,1554
77
77
  novelwriter/assets/syntax/tomorrow_night_eighties.conf,sha256=stO_xH5RzMN0jaZeYr53LVXbpqSmIXALBFGVLlJBDLA,1557
78
- novelwriter/assets/text/credits_en.htm,sha256=NkC68OmhTCGItDWUMhzUcPQ2O03CiIA5LZX0hBRx0YI,3010
78
+ novelwriter/assets/text/credits_en.htm,sha256=MIDbAovnNUt-prTImuC5mDnVTqUknwPtj43tqFL9gPQ,3119
79
79
  novelwriter/assets/text/lipsum.txt,sha256=sGA6AC_p4jrUifxLxy3cSODeRwkpBNXDPGgX6YRTm2s,62937
80
80
  novelwriter/assets/themes/cyberpunk_night.conf,sha256=UT4tm03jBc-JRDrUKb6aLBJVF2JkQl-fha4_iHGpAs4,1331
81
81
  novelwriter/assets/themes/default.conf,sha256=K0LZ64lODpCtUg3pISTAP4UKvEYMvPkmIYN9RDEaWwk,72
@@ -85,13 +85,13 @@ novelwriter/assets/themes/dracula.conf,sha256=c0yNTVaI2vTQIyS1BL6XqJjsnRX0N1sMfa
85
85
  novelwriter/assets/themes/snazzy.conf,sha256=-Qg6-6UxTXyq6pyFCFGNSeGQEVb2pjX9hErCgB58DuE,1381
86
86
  novelwriter/assets/themes/solarized_dark.conf,sha256=17Ylx-hA4zS92MWgWxiTqchUCYVCMvqIwNEiSOJEc-0,1325
87
87
  novelwriter/assets/themes/solarized_light.conf,sha256=zb9M41VnPjR0nx6ZP-YcY-cb-TgS_oIHlok-L3HfJek,1326
88
- novelwriter/core/buildsettings.py,sha256=Tbt5yhY1ijjsD_UIhauJYlNDmX9m6Y22LqztwRUUf8M,25042
88
+ novelwriter/core/buildsettings.py,sha256=9dqXlV6oVYe2UmqSrx54PFqvRYcjOJmJEIaFmqjvaFs,25045
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=lDmRyy-Kjb4gSjaYFj5vAzI40oMQr_xCAtb4XLmOVBQ,40964
93
- novelwriter/core/indexdata.py,sha256=ywfbKZEzK2cKWXepM2M8YoAr-G2qSF9L_GV4GIvQB5s,13177
94
- novelwriter/core/item.py,sha256=-2HQAjjl9fYvgGWp2B5fDeuVb6ZbdU8Tb4QPPUS6Iak,19687
92
+ novelwriter/core/index.py,sha256=Gzg1Co_oNFVx_yBqIN--f-yWGxP6OVH67DsME0-A2qU,41319
93
+ novelwriter/core/indexdata.py,sha256=4FX4DJHfTpJPhiOilPpKVAuj5iDh6OGFI89Z_fQdLE0,13458
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
97
97
  novelwriter/core/options.py,sha256=CTHf2RclRBiZM9No-fZ2x6XyJ1oRPvEoPuujQR8wE0I,7462
@@ -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
- novelwriter/gui/doceditor.py,sha256=w9MjT-puO9Yc-EMYZ0yr5QTTqj4hgwAzcjWcyZFi10A,117508
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,14 @@ 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=B0uMnWEdKG687vB4DTLMmeyorcDmcpSyygUJSbmzxmY,62154
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.7rc1.dist-info/licenses/LICENSE.md,sha256=2GirkkLrPfQqx7fACKRJjtKJUegKc8067erGvcDVQHM,32197
158
- novelwriter-2.7rc1.dist-info/METADATA,sha256=dYfersTN4Deo4Z9Lxx9btCzcEfIPCU6C5V3S0Go5OSw,2555
159
- novelwriter-2.7rc1.dist-info/WHEEL,sha256=Nw36Djuh_5VDukK0H78QzOX-_FQEo6V37m3nkm96gtU,91
160
- novelwriter-2.7rc1.dist-info/entry_points.txt,sha256=YDUG1w361LtLsjD3YhxoTDTwM17JA5-nigjC6j5C74A,45
161
- novelwriter-2.7rc1.dist-info/top_level.txt,sha256=wFFEucjEeNC_Ap5ULBuEutg5a1Uc0-YO9uFT5L2naNI,12
162
- novelwriter-2.7rc1.dist-info/RECORD,,
157
+ novelwriter-2.7.2.dist-info/licenses/LICENSE.md,sha256=2GirkkLrPfQqx7fACKRJjtKJUegKc8067erGvcDVQHM,32197
158
+ novelwriter-2.7.2.dist-info/METADATA,sha256=eCH_hT1p3s1ABiIDhMZqYDHIC2lslcwPM7i75kxlIWs,2458
159
+ novelwriter-2.7.2.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
160
+ novelwriter-2.7.2.dist-info/entry_points.txt,sha256=YDUG1w361LtLsjD3YhxoTDTwM17JA5-nigjC6j5C74A,45
161
+ novelwriter-2.7.2.dist-info/top_level.txt,sha256=wFFEucjEeNC_Ap5ULBuEutg5a1Uc0-YO9uFT5L2naNI,12
162
+ novelwriter-2.7.2.dist-info/RECORD,,
@@ -1,5 +1,5 @@
1
1
  Wheel-Version: 1.0
2
- Generator: setuptools (80.7.1)
2
+ Generator: setuptools (80.9.0)
3
3
  Root-Is-Purelib: true
4
4
  Tag: py3-none-any
5
5