checkpoint
This commit is contained in:
		
							
								
								
									
										75
									
								
								lib/utils.py
									
									
									
									
									
								
							
							
						
						
									
										75
									
								
								lib/utils.py
									
									
									
									
									
								
							| @@ -1,7 +1,9 @@ | ||||
| """Utility Functions.""" | ||||
| from typing import NoReturn | ||||
| from typing import NoReturn, Self | ||||
|  | ||||
| from PyQt6.QtCore import QCoreApplication | ||||
| from PyQt6.QtCore import QCoreApplication, QDir, QStandardPaths, Qt | ||||
| from PyQt6.QtGui import QColor, QFont, QFontDatabase | ||||
| from PyQt6.QtNetwork import QNetworkAccessManager, QNetworkDiskCache | ||||
| from PyQt6.QtSql import QSqlQuery | ||||
|  | ||||
| translate = QCoreApplication.translate | ||||
| @@ -19,3 +21,72 @@ def query_error(query: QSqlQuery) -> NoReturn: | ||||
|         ) | ||||
|     ) | ||||
|     raise Exception(translate("MainWindow", "SQL Error")) | ||||
|  | ||||
| class Resources: | ||||
|     _instance = None | ||||
|     nam = QNetworkAccessManager() | ||||
|     headerFont: QFont | ||||
|     labelFont: QFont | ||||
|     boldFont: QFont | ||||
|     textFont: QFont | ||||
|     italicFont: QFont | ||||
|     capsFont: QFont | ||||
|     smallCapsFont: QFont | ||||
|     phonicFont: QFont | ||||
|  | ||||
|     baseColor: QColor | ||||
|     linkColor: QColor | ||||
|     subduedColor: QColor | ||||
|  | ||||
|     def __new__(cls: type[Self]) -> Self: | ||||
|         if cls._instance: | ||||
|             return cls._instance | ||||
|         cls._instance = super(Resources, cls).__new__(cls) | ||||
|         return cls._instance | ||||
|  | ||||
|     def __init__(self) -> None: | ||||
|         super(Resources, self).__init__() | ||||
|         # | ||||
|         # Fonts | ||||
|         # | ||||
|         self.headerFont = QFontDatabase.font("OpenDyslexic", None, 10) | ||||
|         self.headerFont.setPixelSize(48) | ||||
|         self.labelFont = QFont(self.headerFont) | ||||
|         self.labelFont.setPixelSize(30) | ||||
|         self.boldFont = QFont(self.headerFont) | ||||
|         self.boldFont.setPixelSize(20) | ||||
|         self.textFont = QFont(self.boldFont) | ||||
|         self.italicFont = QFont(self.boldFont) | ||||
|         self.capsFont = QFont(self.boldFont) | ||||
|         self.smallCapsFont = QFont(self.boldFont) | ||||
|  | ||||
|         self.headerFont.setWeight(QFont.Weight.Bold) | ||||
|         self.boldFont.setBold(True) | ||||
|         self.italicFont.setItalic(True) | ||||
|         self.capsFont.setCapitalization(QFont.Capitalization.AllUppercase) | ||||
|         self.smallCapsFont.setCapitalization(QFont.Capitalization.SmallCaps) | ||||
|  | ||||
|         self.phonicFont = QFontDatabase.font("Gentium", None, 10) | ||||
|         self.phonicFont.setPixelSize(20) | ||||
|  | ||||
|         # | ||||
|         # colors | ||||
|         # | ||||
|         self.baseColor = QColor(Qt.GlobalColor.white) | ||||
|         self.linkColor = QColor("#4a7d95") | ||||
|         self.subduedColor = QColor(Qt.GlobalColor.gray) | ||||
|  | ||||
|         # | ||||
|         # Setup the Network Manager | ||||
|         # | ||||
|         cacheDir = QDir( | ||||
|             QStandardPaths.writableLocation( | ||||
|                 QStandardPaths.StandardLocation.GenericCacheLocation | ||||
|             ) | ||||
|         ) | ||||
|         cacheDir.mkdir("Troglodite") | ||||
|         cacheDir = QDir(cacheDir.path() + QDir.separator() + "Troglodite") | ||||
|         netCache = QNetworkDiskCache() | ||||
|         netCache.setCacheDirectory(cacheDir.path()) | ||||
|         self.nam.setCache(netCache) | ||||
|         return | ||||
|   | ||||
							
								
								
									
										142
									
								
								lib/words.py
									
									
									
									
									
								
							
							
						
						
									
										142
									
								
								lib/words.py
									
									
									
									
									
								
							| @@ -2,22 +2,20 @@ import importlib | ||||
| import pkgutil | ||||
| import json | ||||
| import re | ||||
| from typing import Any, Dict, cast | ||||
| from typing import Any, TypedDict, cast | ||||
|  | ||||
| from PyQt6.QtCore import ( | ||||
|     QUrl, | ||||
|     Qt, | ||||
|     pyqtSlot, | ||||
| ) | ||||
| from PyQt6.QtGui import ( | ||||
|     QColor, | ||||
|     QFont, | ||||
|     QFontDatabase, | ||||
| ) | ||||
| from PyQt6.QtNetwork import QNetworkAccessManager | ||||
| from PyQt6.QtSql import QSqlQuery | ||||
| from PyQt6.QtWidgets import QScrollArea | ||||
|  | ||||
| from lib import query_error | ||||
| from lib.utils import query_error, Resources | ||||
| from lib.sounds import SoundOff | ||||
| from lib.definition import Definition, Line, Fragment | ||||
|  | ||||
| @@ -32,20 +30,22 @@ discovered_plugins = { | ||||
|  | ||||
| API = "https://api.dictionaryapi.dev/api/v2/entries/en/{word}" | ||||
|  | ||||
| class WordType(TypedDict): | ||||
|     word: str | ||||
|     source: str | ||||
|     definition: str | ||||
|      | ||||
| class Word: | ||||
|     """All processing of a dictionary word.""" | ||||
|  | ||||
|     _words: dict[str, Any] = {} | ||||
|     _resources: Dict[str, Any] = {} | ||||
|     _nam = QNetworkAccessManager() | ||||
|     _words: dict[str, WordType] = {} | ||||
|      | ||||
|     def __init__(self, word: str) -> None: | ||||
|         Word.set_resources() | ||||
|         # | ||||
|         # Have we already retrieved this word? | ||||
|         # | ||||
|         try: | ||||
|             self.current = json.loads(Word._words[word]) | ||||
|             self.current = Word._words[word] | ||||
|             return | ||||
|         except KeyError: | ||||
|             pass | ||||
| @@ -82,50 +82,6 @@ class Word: | ||||
|             query_error(query) | ||||
|         return | ||||
|  | ||||
|     @classmethod | ||||
|     def set_resources(cls) -> None: | ||||
|         if len(cls._resources.keys()) > 0: | ||||
|             return | ||||
|         # | ||||
|         # Colors we used | ||||
|         # | ||||
|         headerFont = QFontDatabase.font("OpenDyslexic", None, 10) | ||||
|         headerFont.setPixelSize(48) | ||||
|         labelFont = QFont(headerFont) | ||||
|         labelFont.setPixelSize(30) | ||||
|         boldFont = QFont(headerFont) | ||||
|         boldFont.setPixelSize(20) | ||||
|         textFont = QFont(boldFont) | ||||
|         italicFont = QFont(boldFont) | ||||
|         capsFont = QFont(boldFont) | ||||
|         smallCapsFont = QFont(boldFont) | ||||
|  | ||||
|         headerFont.setWeight(QFont.Weight.Bold) | ||||
|         boldFont.setBold(True) | ||||
|         italicFont.setItalic(True) | ||||
|         capsFont.setCapitalization(QFont.Capitalization.AllUppercase) | ||||
|         smallCapsFont.setCapitalization(QFont.Capitalization.SmallCaps) | ||||
|  | ||||
|         phonicFont = QFontDatabase.font("Gentium", None, 10) | ||||
|         phonicFont.setPixelSize(20) | ||||
|  | ||||
|         cls._resources = { | ||||
|             "colors": { | ||||
|                 "base": QColor(Qt.GlobalColor.white), | ||||
|                 "link": QColor("#4a7d95"), | ||||
|                 "subdued": QColor(Qt.GlobalColor.gray), | ||||
|             }, | ||||
|             "fonts": { | ||||
|                 "header": headerFont, | ||||
|                 "label": labelFont, | ||||
|                 "phonic": phonicFont, | ||||
|                 "bold": boldFont, | ||||
|                 "italic": italicFont, | ||||
|                 "text": textFont, | ||||
|                 "caps": capsFont, | ||||
|                 "smallCaps": smallCapsFont, | ||||
|             }, | ||||
|         } | ||||
|     @pyqtSlot() | ||||
|     def playSound(self) -> None: | ||||
|         url = discovered_plugins[self.current['source']].getFirstSound(self.current['definition']) | ||||
| @@ -145,23 +101,15 @@ class Word: | ||||
|             raise Exception(f"Unknown source: {src}") | ||||
|  | ||||
|     def get_def(self) -> list[Line]: | ||||
|         if len(self._lines) > 0: | ||||
|             return self._lines | ||||
|         src = self.current['source'] | ||||
|         try: | ||||
|             return discovered_plugins[src].getDef(self.current) | ||||
|             lines = discovered_plugins[src].getDef(self.current["definition"]) | ||||
|             return lines | ||||
|         except KeyError: | ||||
|             raise Exception(f"Unknown source: {self.current['source']}") | ||||
|  | ||||
|     def mw_def(self) -> list[Line]: | ||||
|         lines: list[Line] = [] | ||||
|         # print(json.dumps(self.current,indent=2)) | ||||
|         for entry in self.current["definition"]: | ||||
|             lines += self.mw_def_entry(entry) | ||||
|         self._lines = lines | ||||
|         return lines | ||||
|  | ||||
|     def mw_seq(self, seq: list[Any]) -> list[Line]: | ||||
|         r=Resources() | ||||
|         lines: list[Line] = [] | ||||
|         outer = " " | ||||
|         inner = " " | ||||
| @@ -189,15 +137,15 @@ class Word: | ||||
|                 line = Line() | ||||
|                 frag = Fragment( | ||||
|                     f"{outer} {inner} ", | ||||
|                     self._resources["fonts"]["bold"], | ||||
|                     color=self._resources["colors"]["base"], | ||||
|                     r.boldFont, | ||||
|                     color=r.baseColor | ||||
|                 ) | ||||
|                 outer = " " | ||||
|                 line.addFragment(frag) | ||||
|                 frag = Fragment( | ||||
|                     text, | ||||
|                     self._resources["fonts"]["italic"], | ||||
|                     color=self._resources["colors"]["base"], | ||||
|                     r.italicFont, | ||||
|                     color=r.baseColor | ||||
|                 ) | ||||
|                 frag.setLeft(30) | ||||
|                 line.addFragment(frag) | ||||
| @@ -209,16 +157,16 @@ class Word: | ||||
|                         line = Line() | ||||
|                         frag = Fragment( | ||||
|                             f"{outer} {inner} ", | ||||
|                             self._resources["fonts"]["bold"], | ||||
|                             color=self._resources["colors"]["base"], | ||||
|                             r.boldFont, | ||||
|                             color=r.baseColor | ||||
|                         ) | ||||
|                         outer = " " | ||||
|                         frag.setLeft(10) | ||||
|                         line.addFragment(frag) | ||||
|                         frag = Fragment( | ||||
|                             dt[1], | ||||
|                             self._resources["fonts"]["text"], | ||||
|                             color=self._resources["colors"]["base"], | ||||
|                             r.textFont, | ||||
|                             color=r.baseColor | ||||
|                         ) | ||||
|                         frag.setLeft(30) | ||||
|                         line.addFragment(frag) | ||||
| @@ -228,14 +176,14 @@ class Word: | ||||
|                             line = Line() | ||||
|                             frag = Fragment( | ||||
|                                 f"  ", | ||||
|                                 self._resources["fonts"]["bold"], | ||||
|                                 r.boldFont | ||||
|                             ) | ||||
|                             frag.setLeft(45) | ||||
|                             line.addFragment(frag) | ||||
|                             line.addFragment( | ||||
|                                 Fragment( | ||||
|                                     vis["t"], | ||||
|                                     self._resources["fonts"]["text"], | ||||
|                                     r.textFont, | ||||
|                                     color=QColor("#aaa"), | ||||
|                                 ) | ||||
|                             ) | ||||
| @@ -250,8 +198,8 @@ class Word: | ||||
|                                         line = Line() | ||||
|                                     frag = Fragment( | ||||
|                                         "\u27F6 " + seg[1], | ||||
|                                         self._resources["fonts"]["text"], | ||||
|                                         color=self._resources["colors"]["base"], | ||||
|                                         r.textFont, | ||||
|                                         color=r.baseColor | ||||
|                                     ) | ||||
|                                     frag.setLeft(30) | ||||
|                                     line.addFragment(frag) | ||||
| @@ -265,19 +213,17 @@ class Word: | ||||
|         return lines | ||||
|  | ||||
|     def mw_def_entry(self, entry: dict[str, Any]) -> list[Line]: | ||||
|         r = Resources() | ||||
|         # | ||||
|         # Easy reference to colors | ||||
|         # | ||||
|         base = self._resources["colors"]["base"] | ||||
|         blue = self._resources["colors"]["blue"] | ||||
|  | ||||
|         lines: list[Line] = [] | ||||
|         line = Line() | ||||
|         hw = re.sub(r"\*", "", entry["hwi"]["hw"]) | ||||
|         frag = Fragment(hw, self._resources["fonts"]["header"], color=base) | ||||
|         frag = Fragment(hw, r.headerFont, color=r.baseColor) | ||||
|         line.addFragment(frag) | ||||
|         frag = Fragment( | ||||
|             " " + entry["fl"], self._resources["fonts"]["label"], color=blue | ||||
|             " " + entry["fl"], r.labelFont, color=r.linkColor | ||||
|         ) | ||||
|         line.addFragment(frag) | ||||
|         lines.append(line) | ||||
| @@ -288,8 +234,8 @@ class Word: | ||||
|             for vrs in entry["vrs"]: | ||||
|                 frag = Fragment( | ||||
|                     space + vrs["va"], | ||||
|                     self._resources["fonts"]["label"], | ||||
|                     color=base, | ||||
|                     r.labelFont, | ||||
|                     color=r.baseColor | ||||
|                 ) | ||||
|                 space = " " | ||||
|                 line.addFragment(frag) | ||||
| @@ -298,16 +244,16 @@ class Word: | ||||
|             line = Line() | ||||
|             frag = Fragment( | ||||
|                 entry["hwi"]["hw"] + " ", | ||||
|                 self._resources["fonts"]["phonic"], | ||||
|                 color=base, | ||||
|                 r.phonicFont, | ||||
|                 color=r.baseColor, | ||||
|             ) | ||||
|             line.addFragment(frag) | ||||
|             for prs in entry["hwi"]["prs"]: | ||||
|                 audio = self.mw_sound_url(prs) | ||||
|                 audio = None | ||||
|                 if audio is None: | ||||
|                     audio = "" | ||||
|                 frag = Fragment( | ||||
|                     prs["mw"], self._resources["fonts"]["phonic"], color=blue | ||||
|                     prs["mw"], r.phonicFont, color=r.linkColor | ||||
|                 ) | ||||
|                 frag.setAudio(audio) | ||||
|                 line.addFragment(frag) | ||||
| @@ -318,7 +264,7 @@ class Word: | ||||
|             for ins in entry["ins"]: | ||||
|                 try: | ||||
|                     frag = Fragment( | ||||
|                         ins["il"], self._resources["fonts"]["text"], color=base | ||||
|                         ins["il"], r.textFont, color=r.baseColor | ||||
|                     ) | ||||
|                     line.addFragment(frag) | ||||
|                     space = " " | ||||
| @@ -326,8 +272,8 @@ class Word: | ||||
|                     pass | ||||
|                 frag = Fragment( | ||||
|                     space + ins["if"], | ||||
|                     self._resources["fonts"]["bold"], | ||||
|                     color=base, | ||||
|                     r.boldFont, | ||||
|                     color=r.baseColor | ||||
|                 ) | ||||
|                 line.addFragment(frag) | ||||
|                 space = "; " | ||||
| @@ -336,8 +282,8 @@ class Word: | ||||
|             line = Line() | ||||
|             frag = Fragment( | ||||
|                 "; ".join(entry["lbs"]), | ||||
|                 self._resources["fonts"]["bold"], | ||||
|                 color=base, | ||||
|                 r.boldFont, | ||||
|                 color=r.baseColor | ||||
|             ) | ||||
|             line.addFragment(frag) | ||||
|             lines.append(line) | ||||
| @@ -345,13 +291,13 @@ class Word: | ||||
|             for k, v in value.items(): | ||||
|                 if k == "sseq":  # has multiple 'senses' | ||||
|                     for seq in v: | ||||
|                         r = self.mw_seq(seq) | ||||
|                         lines += r | ||||
|                         rr = self.mw_seq(seq) | ||||
|                         lines += rr | ||||
|                 elif k == "vd": | ||||
|                     line = Line() | ||||
|                     line.addFragment( | ||||
|                         Fragment( | ||||
|                             v, self._resources["fonts"]["italic"], color=blue | ||||
|                             v, r.italicFont, color=r.linkColor | ||||
|                         ) | ||||
|                     ) | ||||
|                     lines.append(line) | ||||
| @@ -361,7 +307,7 @@ class Word: | ||||
|         # | ||||
|         # Create the header, base word and its label | ||||
|         # | ||||
|         word = self.current["hwi"]["hw"] | ||||
|         word = self.current['definition']["hwi"]["hw"] | ||||
|         label = self.current["fl"] | ||||
|         html = f'<h1 class="def-word">{word} <span class="def-label">{label}</span></h1>\n' | ||||
|  | ||||
| @@ -383,7 +329,7 @@ class Word: | ||||
|         if "prs" in self.current["hwi"].keys(): | ||||
|             tmp = [] | ||||
|             for prs in self.current["hwi"]["prs"]: | ||||
|                 url = self.mw_sound_url(prs) | ||||
|                 url = QUrl() | ||||
|                 how = prs["mw"] | ||||
|                 if url: | ||||
|                     tmp.append(f'<a href="{url}">\\{how}\\</a>') | ||||
|   | ||||
| @@ -1,12 +1,12 @@ | ||||
| from PyQt6.QtGui import QColor | ||||
| from trycast import trycast | ||||
| import json | ||||
| import re | ||||
| from typing import Any, Literal, NamedTuple, NotRequired, TypedDict, cast | ||||
| from typing import Any, NamedTuple, NotRequired, TypedDict | ||||
|  | ||||
| from PyQt6.QtCore import QEventLoop, QUrl, Qt | ||||
| from PyQt6.QtGui import QColor, QFont | ||||
| from PyQt6.QtNetwork import QNetworkRequest | ||||
| from lib.words import Word | ||||
| from lib.utils import Resources | ||||
| from lib.definition import Line, Fragment | ||||
|  | ||||
| registration = { | ||||
| @@ -27,10 +27,6 @@ class VerbalIllustration(TypedDict): | ||||
|     t: str | ||||
|     aq: str | ||||
|      | ||||
| class VerbalIllustrationTuple(NamedTuple): | ||||
|     type_: str # 'vis' | ||||
|     data: list[VerbalIllustration] | ||||
|      | ||||
| class Sound(TypedDict): | ||||
|     audio: str | ||||
|     ref: str | ||||
| @@ -38,12 +34,10 @@ class Sound(TypedDict): | ||||
|  | ||||
| class Pronunciation(TypedDict): | ||||
|     mw: str | ||||
|     l: str | ||||
|     l2: str | ||||
|     pun: str | ||||
|     sound: Sound | ||||
|  | ||||
|  | ||||
|     l: NotRequired[str] | ||||
|     l2: NotRequired[str] | ||||
|     pun: NotRequired[str] | ||||
|     sound: NotRequired[Sound] | ||||
|  | ||||
| class Meta(TypedDict): | ||||
|     id: str | ||||
| @@ -56,12 +50,12 @@ class Meta(TypedDict): | ||||
|  | ||||
| class HeadWordInfo(TypedDict): | ||||
|     hw: str | ||||
|     prs: list[Pronunciation] | ||||
|     prs: NotRequired[list[Pronunciation]] | ||||
|      | ||||
| class HeadWord(TypedDict): | ||||
|     hw: str | ||||
|     prs: list[Pronunciation] | ||||
|     psl: str | ||||
|     prs: NotRequired[list[Pronunciation]] | ||||
|     psl: NotRequired[str] | ||||
|  | ||||
| class Variant(TypedDict): | ||||
|     va: str | ||||
| @@ -109,34 +103,26 @@ class RunInWrap(TypedDict): | ||||
|     text: str | ||||
|     vrs: list[Variant] | ||||
|      | ||||
| class Sense: | ||||
|     dt: list[str] # not full | ||||
|     et: list[str] # not full  | ||||
|     ins: list[Inflection] | ||||
|     lbs: list[str] | ||||
|     prs: list[Pronunciation] | ||||
|     sdsense: DividedSense | ||||
|     sgram: str | ||||
|     sls: list[str] | ||||
|     sn: str | ||||
|     vrs: list[Variant] | ||||
|      | ||||
| class SenseSequence(TypedDict): | ||||
|     sense: Sense | ||||
|     sen: Sense | ||||
| class Sense(TypedDict): | ||||
|     dt: list[list] # not full | ||||
|     et: NotRequired[list[str]] | ||||
|     ins: NotRequired[list[Inflection]] | ||||
|     lbs: NotRequired[list[str]] | ||||
|     prs: NotRequired[list[Pronunciation]] | ||||
|     sdsense: NotRequired[DividedSense] | ||||
|     sgram: NotRequired[str] | ||||
|     sls: NotRequired[list[str]] | ||||
|     sn: NotRequired[str] | ||||
|     vrs: NotRequired[list[Variant]] | ||||
|      | ||||
| class Definition(TypedDict): | ||||
|     sseq: list[SenseSequence] | ||||
|     vd: str | ||||
|     sseq: list[list[list[Any]]] | ||||
|     vd: NotRequired[str] | ||||
|      | ||||
| class Pair(TypedDict): | ||||
|     objType: str | ||||
|     obj: list[Sense]|Sense|str|list[VerbalIllustration]|list[Any] | ||||
|      | ||||
| class EntryX(TypedDict): | ||||
|     meta: Meta | ||||
|     hom: NotRequired[str] | ||||
|     hwi: HeadWordInfo | ||||
|     ahws: NotRequired[list[HeadWord]] | ||||
|     vrs: NotRequired[list[Variant]] | ||||
|     fl: str | ||||
|     def_: list[Definition] | ||||
| Entry = TypedDict( | ||||
|     'Entry', | ||||
|     { | ||||
| @@ -149,13 +135,29 @@ Entry = TypedDict( | ||||
|         'def': list[Definition], | ||||
|     } | ||||
| ) | ||||
| class WordType(TypedDict): | ||||
|     word: str | ||||
|     source: str | ||||
|     definition: dict[str, Any] | ||||
|      | ||||
| def fetch(word:str) ->  dict[str, Any]: | ||||
| def make_pairs(src: list[Any]) -> list[Pair]: | ||||
|     result:list[Pair] = [] | ||||
|     iters = [iter(src)]*2 | ||||
|     for entry in zip(*iters): | ||||
|         pair = { 'objType': entry[0], | ||||
|                  'obj': entry[1], | ||||
|                 } | ||||
|         pair = trycast(Pair, pair) | ||||
|         assert pair is not None | ||||
|         result.append(pair) | ||||
|     return result | ||||
|  | ||||
| def fetch(word:str) ->  WordType: | ||||
|     request = QNetworkRequest() | ||||
|     url = QUrl(API.format(word=word, key=key)) | ||||
|     request.setUrl(url) | ||||
|     request.setTransferTimeout(3000) | ||||
|     reply = Word._nam.get(request) | ||||
|     reply = Resources.nam.get(request) | ||||
|     assert reply is not None | ||||
|     loop = QEventLoop() | ||||
|     reply.finished.connect(loop.quit) | ||||
| @@ -195,16 +197,16 @@ def getFirstSound(definition: list[Entry]) -> QUrl: | ||||
|                             return url | ||||
|     return QUrl() | ||||
|  | ||||
| def do_prs(prs: list[Pronunciation]) -> list[Fragment]: | ||||
| def do_prs(hwi: HeadWordInfo) -> list[Fragment]: | ||||
|     r = Resources() | ||||
|     frags: list[Fragment] = [] | ||||
|     font = trycast(QFont, Word._resources['fonts']['label']) | ||||
|     assert font is not None | ||||
|     linkColor = trycast(QColor, Word._resources['colors']['link']) | ||||
|     assert linkColor is not None | ||||
|     subduedColor = trycast(QColor, Word._resources['colors']['subdued']) | ||||
|     assert subduedColor is not None | ||||
|     font = r.labelFont | ||||
|     linkColor = r.linkColor | ||||
|     subduedColor = r.subduedColor | ||||
|  | ||||
|     for pr in prs: | ||||
|     if 'prs' not in hwi: | ||||
|         return [] | ||||
|     for pr in hwi['prs']: | ||||
|         if 'pun' in pr: | ||||
|             pun = pr['pun'] | ||||
|         else: | ||||
| @@ -216,6 +218,7 @@ def do_prs(prs: list[Pronunciation]) -> list[Fragment]: | ||||
|         frag = Fragment(pr['mw'], font, color=subduedColor) | ||||
|         if 'sound' in pr: | ||||
|             frag.setAudio(soundUrl(pr['sound'])) | ||||
|             frag.setColor(linkColor) | ||||
|         frags.append(frag) | ||||
|         if 'l2' in pr: | ||||
|             frags.append( | ||||
| @@ -223,38 +226,141 @@ def do_prs(prs: list[Pronunciation]) -> list[Fragment]: | ||||
|             ) | ||||
|     return frags | ||||
|  | ||||
| def do_sense(sense: Sense|None) -> tuple[list[Fragment], list[Line]]: | ||||
|     if sense is None: | ||||
|         return ([],[]) | ||||
|     lines: list[Line] = [] | ||||
|     frags: list[Fragment] = [] | ||||
|     r = Resources() | ||||
|     if 'sn' in sense: | ||||
|         sn = sense['sn'] | ||||
|     else: | ||||
|         sn = '' | ||||
|     print(f'{sn}\n\n',json.dumps(sense['dt'], indent=2)) | ||||
|     iters = [iter(sense['dt'])]*2 | ||||
|     for pair in zip(*iters): | ||||
|         pair = trycast(tuple[str, Any], pair) | ||||
|         assert pair is not None | ||||
|         print(pair[0]) | ||||
|         if pair[0] == 'text': | ||||
|             line = Line() | ||||
|             line.addFragment( | ||||
|                 Fragment(pair[1], r.textFont, color=r.baseColor) | ||||
|             ) | ||||
|             lines.append(line) | ||||
|     return (frags, lines) | ||||
|  | ||||
| def do_pseq(outer: int, | ||||
|             inner: int, | ||||
|             pseq: list[list[Pair]]| None ) -> tuple[list[Fragment], list[Line]]: | ||||
|     assert pseq is not None | ||||
|     lines: list[Line] = [] | ||||
|     frags: list[Fragment] = [] | ||||
|     for entry in pseq: | ||||
|         pairs = make_pairs(entry) | ||||
|         for pair in pairs: | ||||
|             if pair['objType'] == 'bs': | ||||
|                 (newFrags, newLines) = do_sense(trycast(Sense, pair['obj'])) | ||||
|                 frags += newFrags | ||||
|                 lines += newLines | ||||
|             elif pair['objType'] == 'sense': | ||||
|                 (newFrags, newLines) = do_sense(trycast(Sense, pair['obj'])) | ||||
|                 frags += newFrags | ||||
|                 lines += newLines | ||||
|             else: | ||||
|                 raise Exception(f"Unknown object type {pair['objType']}") | ||||
|     return (frags, lines) | ||||
|  | ||||
| def do_sseq(sseq:list[list[list[Pair]]]) -> list[Line]: | ||||
|     lines: list[Line] = [] | ||||
|     r = Resources() | ||||
|     for outer, item_o in enumerate(sseq): | ||||
|         line = Line() | ||||
|         line.addFragment( | ||||
|             Fragment(str(outer+1), r.boldFont, color=r.baseColor) | ||||
|         ) | ||||
|         for inner, item_i in enumerate(item_o): | ||||
|             line.addFragment( | ||||
|                 Fragment(chr(ord('a')+inner), r.boldFont, color=r.baseColor) | ||||
|             ) | ||||
|             pairs = make_pairs(item_i) | ||||
|             for pair in pairs: | ||||
|                 objType = pair['objType'] | ||||
|                 if objType == 'sense': | ||||
|                     sense = trycast(Sense, pair['obj']) | ||||
|                     (frags, newlines) = do_sense(sense) | ||||
|                     for frag in frags: | ||||
|                         line.addFragment(frag) | ||||
|                     lines.append(line) | ||||
|                     lines += newlines | ||||
|                 elif objType == 'sen': | ||||
|                     raise Exception(f"sen unimplimented") | ||||
|                 elif objType == 'pseq': | ||||
|                     pseq = trycast(list[list[Pair]], pair['obj']) | ||||
|                     (frags, newlines) = do_pseq(inner, outer, trycast(list[list[Pair]], pair['obj'])) | ||||
|                     for frag in frags: | ||||
|                         line.addFragment(frag) | ||||
|                     lines.append(line) | ||||
|                     lines += newlines | ||||
|                 elif objType == 'bs': | ||||
|                     raise Exception(f"bs unimplimented") | ||||
|                 else: | ||||
|                     raise Exception(f"Unknown object[{objType}] for \n{json.dumps(pair['obj'],indent=2)}") | ||||
|     return lines | ||||
|  | ||||
| def do_def(entry: Definition) -> list[Line]: | ||||
|     r = Resources() | ||||
|     lines: list[Line] = [] | ||||
|     assert trycast(Definition, entry) is not None | ||||
|     if 'vd' in entry: | ||||
|         line = Line() | ||||
|         line.addFragment( | ||||
|             Fragment(entry['vd'], r.italicFont, color = r.linkColor) | ||||
|         ) | ||||
|         lines.append(line) | ||||
|     # | ||||
|     # sseg is required | ||||
|     # | ||||
|     sseq = entry['sseq'] | ||||
|     lines += do_sseq(sseq) | ||||
|     return lines | ||||
|  | ||||
| def getDef(definition: list[Entry]) -> list[Line]: | ||||
|     lines = [] | ||||
|     r = Resources() | ||||
|     lines:list[Line] = [] | ||||
|     # | ||||
|     # Pull the fonts for ease of use | ||||
|     # | ||||
|     headerFont = trycast(QFont, Word._resources['fonts']['header']) | ||||
|     assert headerFont is not None | ||||
|     textFont = trycast(QFont, Word._resources['fonts']['text']) | ||||
|     assert textFont is not None | ||||
|     labelFont = trycast(QFont, Word._resources['fonts']['label']) | ||||
|     assert labelFont is not None | ||||
|     headerFont = r.headerFont | ||||
|     textFont = r.textFont | ||||
|     labelFont = r.labelFont | ||||
|     # | ||||
|     # Pull the colors for ease of use | ||||
|     # | ||||
|     baseColor = trycast(QColor, Word._resources['colors']['base']) | ||||
|     assert baseColor is not None | ||||
|     linkColor = trycast(QColor, Word._resources['colors']['link']) | ||||
|     assert linkColor is not None | ||||
|     subduedColor = trycast(QColor, Word._resources['colors']['subdued']) | ||||
|     assert subduedColor is not None | ||||
|     baseColor = r.baseColor | ||||
|     linkColor = r.linkColor | ||||
|     subduedColor = r.subduedColor | ||||
|  | ||||
|     # | ||||
|     # No need to figure it out each time it is used | ||||
|     # | ||||
|     entries = 0 | ||||
|     id = definition[0]['meta']['id'] | ||||
|     id = ':'.split(id)[0].lower() | ||||
|     id = definition[0]['meta']['id'].lower().split(':')[0] | ||||
|     uses: dict[str,int] = {} | ||||
|     for entry in definition: | ||||
|         if entry['meta']['id'].lower() == id: | ||||
|         testId = entry['meta']['id'].lower().split(':')[0] | ||||
|         if testId == id: | ||||
|             entries += 1 | ||||
|             try: | ||||
|                 uses[entry['fl']] = uses.get(entry['fl'], 0) + 1 | ||||
|             except KeyError: | ||||
|                 pass | ||||
|     used: dict[str, int] = {} | ||||
|     for k in uses.keys(): | ||||
|         used[k] = 0 | ||||
|     for count, entry in enumerate(definition): | ||||
|         if entry['meta']['id'].lower() != id: | ||||
|         testId = entry['meta']['id'].lower().split(':')[0] | ||||
|         if testId != id: | ||||
|             continue | ||||
|         # | ||||
|         # Create the First line from the hwi, [ahws] and fl | ||||
| @@ -270,13 +376,16 @@ def getDef(definition: list[Entry]) -> list[Line]: | ||||
|             for ahw in ahws: | ||||
|                 hw = re.sub(r'\*', '', ahw['hw']) | ||||
|                 line.addFragment(Fragment(', ' + hw, headerFont, color=baseColor)) | ||||
|         if 'hom' in entry: | ||||
|              | ||||
|         if  entries > 1: | ||||
|             frag = Fragment(f" {count + 1} of {entries} ", textFont, color= subduedColor) | ||||
|             frag.setBackground(QColor(Qt.GlobalColor.gray)) | ||||
|             line.addFragment(frag) | ||||
|         if 'fl' in entry: | ||||
|             frag = Fragment(f"{count} of {entries} ", textFont, color= | ||||
|         frag.setBackground(QColor(Qt.GlobalColor.gray)) | ||||
|         line.addFragment(frag) | ||||
|         line.addFragment(Fragment(entry['fl'], labelFont, color=baseColor)) | ||||
|             text = entry['fl'] | ||||
|             used[text] += 1 | ||||
|             if uses[text] > 1: | ||||
|                 text += f' ({used[text]})' | ||||
|             line.addFragment(Fragment(text, labelFont, color=baseColor)) | ||||
|         lines.append(line) | ||||
|  | ||||
|         # | ||||
| @@ -284,11 +393,15 @@ def getDef(definition: list[Entry]) -> list[Line]: | ||||
|         # While 'prs' is optional, the headword is not.  This gets us what we want. | ||||
|         # | ||||
|         line = Line() | ||||
|         hw = re.sub(r'\*', '\u00b7', hwi['hw']) | ||||
|         line.addFragment(Fragment(hw + ' ', textFont, color=subduedColor)) | ||||
|         for frag in do_prs(hwi['prs']): | ||||
|         if hwi['hw'].find('*') >= 0: | ||||
|             hw = re.sub(r'\*', '\u00b7', hwi['hw']) | ||||
|             line.addFragment(Fragment(hw + ' ', textFont, color=subduedColor)) | ||||
|         for frag in do_prs(hwi): | ||||
|             line.addFragment(frag) | ||||
|  | ||||
|         # | ||||
|         # Try for  | ||||
|     return [Line()] | ||||
|         if len(line.getLine()) > 0: | ||||
|             lines.append(line) | ||||
|         defines = trycast(list[Definition], entry['def']) | ||||
|         assert defines is not None | ||||
|         for define in defines: | ||||
|             lines += do_def(define) | ||||
|     return lines | ||||
|   | ||||
		Reference in New Issue
	
	Block a user