"""Base classes and other objects used by enumerations.""" from __future__ import annotations import enum import textwrap from typing import TYPE_CHECKING, Any, Dict, Type, TypeVar if TYPE_CHECKING: from typing_extensions import Self _T = TypeVar("_T", bound="BaseXmlEnum") class BaseEnum(int, enum.Enum): """Base class for Enums that do not map XML attr values. The enum's value will be an integer, corresponding to the integer assigned the corresponding member in the MS API enum of the same name. """ def __new__(cls, ms_api_value: int, docstr: str): self = int.__new__(cls, ms_api_value) self._value_ = ms_api_value self.__doc__ = docstr.strip() return self def __str__(self): """The symbolic name and string value of this member, e.g. 'MIDDLE (3)'.""" return f"{self.name} ({self.value})" class BaseXmlEnum(int, enum.Enum): """Base class for Enums that also map XML attr values. The enum's value will be an integer, corresponding to the integer assigned the corresponding member in the MS API enum of the same name. """ xml_value: str def __new__(cls, ms_api_value: int, xml_value: str, docstr: str): self = int.__new__(cls, ms_api_value) self._value_ = ms_api_value self.xml_value = xml_value self.__doc__ = docstr.strip() return self def __str__(self): """The symbolic name and string value of this member, e.g. 'MIDDLE (3)'.""" return f"{self.name} ({self.value})" @classmethod def from_xml(cls, xml_value: str | None) -> Self: """Enumeration member corresponding to XML attribute value `xml_value`. Example:: >>> WD_PARAGRAPH_ALIGNMENT.from_xml("center") WD_PARAGRAPH_ALIGNMENT.CENTER """ member = next((member for member in cls if member.xml_value == xml_value), None) if member is None: raise ValueError(f"{cls.__name__} has no XML mapping for '{xml_value}'") return member @classmethod def to_xml(cls: Type[_T], value: int | _T | None) -> str | None: """XML value of this enum member, generally an XML attribute value.""" # -- presence of multi-arg `__new__()` method fools type-checker, but getting a # -- member by its value using EnumCls(val) works as usual. return cls(value).xml_value class DocsPageFormatter: """Generate an .rst doc page for an enumeration. Formats a RestructuredText documention page (string) for the enumeration class parts passed to the constructor. An immutable one-shot service object. """ def __init__(self, clsname: str, clsdict: Dict[str, Any]): self._clsname = clsname self._clsdict = clsdict @property def page_str(self): """The RestructuredText documentation page for the enumeration. This is the only API member for the class. """ tmpl = ".. _%s:\n\n%s\n\n%s\n\n----\n\n%s" components = ( self._ms_name, self._page_title, self._intro_text, self._member_defs, ) return tmpl % components @property def _intro_text(self): """Docstring of the enumeration, formatted for documentation page.""" try: cls_docstring = self._clsdict["__doc__"] except KeyError: cls_docstring = "" if cls_docstring is None: return "" return textwrap.dedent(cls_docstring).strip() def _member_def(self, member: BaseEnum | BaseXmlEnum): """Return an individual member definition formatted as an RST glossary entry, wrapped to fit within 78 columns.""" assert member.__doc__ is not None member_docstring = textwrap.dedent(member.__doc__).strip() member_docstring = textwrap.fill( member_docstring, width=78, initial_indent=" " * 4, subsequent_indent=" " * 4, ) return "%s\n%s\n" % (member.name, member_docstring) @property def _member_defs(self): """A single string containing the aggregated member definitions section of the documentation page.""" members = self._clsdict["__members__"] member_defs = [self._member_def(member) for member in members if member.name is not None] return "\n".join(member_defs) @property def _ms_name(self): """The Microsoft API name for this enumeration.""" return self._clsdict["__ms_name__"] @property def _page_title(self): """The title for the documentation page, formatted as code (surrounded in double-backtics) and underlined with '=' characters.""" title_underscore = "=" * (len(self._clsname) + 4) return "``%s``\n%s" % (self._clsname, title_underscore)