Skip to content

File property.py

File List > mkdoxy > property.py

Go to the documentation of this file

import re
from xml.etree.ElementTree import Element as Element

from mkdoxy.constants import Kind
from mkdoxy.markdown import escape
from mkdoxy.xml_parser import XmlParser


class Property:
    class Details:
        def __init__(self, xml: Element, parser: XmlParser, kind: Kind):
            self.xmlxml = xml
            self.parserparser = parser
            self.kindkind = kind

        def md(self, plain: bool = False) -> str:
            detaileddescription = self.xmlxml.find("detaileddescription")
            if list(detaileddescription):
                return self.parserparser.paras_as_str(detaileddescription, plain=plain)
            else:
                return ""

        def plain(self) -> str:
            return self.mdmd(plain=True)

        def has(self) -> bool:
            detaileddescription = self.xmlxml.find("detaileddescription")
            return len(list(detaileddescription)) > 0

    class Brief:
        def __init__(self, xml: Element, parser: XmlParser, kind: Kind):
            self.xmlxml = xml
            self.parserparser = parser
            self.kindkind = kind

        def md(self, plain: bool = False) -> str:
            briefdescription = self.xmlxml.find("briefdescription")
            if briefdescription is None:
                return ""

            paras = briefdescription.findall("para")
            if len(paras) > 0:
                text = [self.parserparser.paras_as_str(para, italic=True, plain=plain) for para in paras]
                return " ".join(text)
            else:
                return ""

        def plain(self) -> str:
            return self.mdmd(plain=True)

        def has(self) -> bool:
            detaileddescription = self.xmlxml.find("briefdescription")
            return len(list(detaileddescription)) > 0

    class Includes:
        def __init__(self, xml: Element, parser: XmlParser, kind: Kind):
            self.xmlxml = xml
            self.parserparser = parser
            self.kindkind = kind

        def md(self, plain: bool = False) -> [str]:
            return self.arrayarray(plain=False)

        def plain(self) -> [str]:
            return self.arrayarray(plain=True)

        def array(self, plain: bool = False) -> [str]:
            ret = []
            for includes in self.xmlxml.findall("includes"):
                incl = includes.text if plain else self.parserparser.reference_as_str(includes)
                if includes.get("local") == "yes":
                    ret.append(f'"{incl}"')
                elif plain:
                    ret.append(f"<{incl}>")
                else:
                    ret.append(f"&lt;{incl}&gt;")
            return ret

        def has(self) -> bool:
            return len(self.xmlxml.findall("includes")) > 0

    class Type:
        def __init__(self, xml: Element, parser: XmlParser, kind: Kind):
            self.xmlxml = xml
            self.parserparser = parser
            self.kindkind = kind

        def md(self, plain: bool = False) -> str:
            para = self.xmlxml.find("type")
            return self.parserparser.paras_as_str(para, plain=plain) if para is not None else ""

        def plain(self) -> str:
            return self.mdmd(plain=True)

        def has(self) -> bool:
            return self.xmlxml.find("type") is not None

    class Location:
        def __init__(self, xml: Element, parser: XmlParser, kind: Kind):
            self.xmlxml = xml
            self.parserparser = parser
            self.kindkind = kind

        def md(self, plain: bool = False) -> str:
            return self.plainplain()

        def plain(self) -> str:
            loc = self.xmlxml.find("location")
            return loc.get("file") if loc is not None else ""

        def line(self) -> int:
            loc = self.xmlxml.find("location")
            return int(loc.get("line")) if loc is not None else 0

        def column(self) -> int:
            loc = self.xmlxml.find("location")
            return int(loc.get("column")) if loc is not None else 0

        def bodystart(self) -> int:
            loc = self.xmlxml.find("location")
            return int(loc.get("bodystart")) if loc is not None else 0

        def bodyend(self) -> int:
            loc = self.xmlxml.find("location")
            return int(loc.get("bodyend")) if loc is not None else 0

        def has(self) -> bool:
            return self.xmlxml.find("location") is not None

    class Params:
        def __init__(self, xml: Element, parser: XmlParser, kind: Kind):
            self.xmlxml = xml
            self.parserparser = parser
            self.kindkind = kind

        def md(self, plain: bool = False) -> str:
            return ", ".join(self.arrayarray(plain=plain))

        def plain(self) -> str:
            return ", ".join(self.arrayarray(plain=True))

        def array(self, plain: bool = False) -> [str]:
            ret = []
            for param in self.xmlxml.findall("param"):
                p = ""
                type = param.find("type")
                p = self.parserparser.paras_as_str(type, plain=plain)

                declname = param.find("declname")
                if declname is not None:
                    p += f" {self.parser.paras_as_str(declname, plain=plain)}"
                else:
                    defname = param.find("defname")
                    if defname is not None:
                        p += f" {self.parser.paras_as_str(defname, plain=plain)}"

                defval = param.find("defval")
                if defval is not None:
                    p += f"={self.parser.paras_as_str(defval, plain=plain)}"

                ret.append(p.strip())
            return ret

        def has(self) -> bool:
            return len(self.xmlxml.findall("param")) > 0

    class TemplateParams:
        def __init__(self, xml: Element, parser: XmlParser, kind: Kind):
            self.xmlxml = xml
            self.parserparser = parser
            self.kindkind = kind

        def md(self, plain: bool = False) -> str:
            return ", ".join(self.arrayarray(plain=plain))

        def plain(self) -> str:
            return ", ".join(self.arrayarray(plain=True))

        def array(self, plain: bool = False, notype: bool = False) -> [str]:
            ret = []
            templateparamlist = self.xmlxml.find("templateparamlist")
            if templateparamlist is not None:
                for param in templateparamlist.findall("param"):
                    if notype:
                        declname = param.find("declname")
                        if declname is None:
                            declname = param.find("type")
                        ret.append(self.parserparser.paras_as_str(declname, plain=plain))
                    else:
                        type = param.find("type")
                        declaration = self.parserparser.paras_as_str(type, plain=plain)
                        declname = param.find("declname")
                        if declname is not None:
                            declaration += f" {self.parser.paras_as_str(declname, plain=plain)}"
                        ret.append(declaration)
            return ret

        def has(self) -> bool:
            return self.xmlxml.find("templateparamlist") is not None

    class CodeBlock:
        def __init__(self, xml: Element, parser: XmlParser, kind: Kind):
            self.xmlxml = xml
            self.parserparser = parser
            self.kindkind = kind

        def md(self, plain: bool = False) -> str:
            return self.plainplain()

        def plain(self) -> str:
            return ""

        def has(self) -> bool:
            return True

    class Specifiers:
        def __init__(self, xml: Element, parser: XmlParser, kind: Kind):
            self.xmlxml = xml
            self.parserparser = parser
            self.kindkind = kind

        def md(self, plain: bool = False) -> str:
            return self.parsedparsed()

        def plain(self) -> str:
            argss = self.xmlxml.find("argsstring")
            return "" if argss is None or argss.text is None else argss.text

        def parsed(self) -> str:
            argss = self.xmlxml.find("argsstring")
            if argss is None or argss.text is None:
                return ""

            argsstring = argss.text
            ret = []

            # Is deleted?
            if re.search("\\)\\s*=\\s*delete", argsstring):
                ret.append("= delete")

            # Is default?
            if re.search("\\)\\s*=\\s*default", argsstring):
                ret.append("= default")

            # Is noexcept
            if re.search("\\).*noexcept", argsstring):
                ret.append("noexcept")

            # Is override
            if re.search("\\).*override", argsstring):
                ret.append("override")

            # Is const?
            if self.xmlxml.get("const") == "yes":
                ret.append("const")

            # Is pure?
            virt = self.xmlxml.get("virt")
            if virt == "pure-virtual":
                ret.append("= 0")

            return " ".join(ret)

        def has(self) -> bool:
            return self.xmlxml.find("argsstring") is not None

    class Values:
        def __init__(self, xml: Element, parser: XmlParser, kind: Kind):
            self.xmlxml = xml
            self.parserparser = parser
            self.kindkind = kind

        def md(self, plain: bool = False) -> str:
            return ", ".join(self.arrayarray(plain=plain))

        def plain(self) -> str:
            return ", ".join(self.arrayarray(plain=False))

        def array(self, plain: bool = False) -> [str]:
            ret = []
            if self.kindkind.is_enum():
                for enumvalue in self.xmlxml.findall("enumvalue"):
                    p = "**" + escape(enumvalue.find("name").text) + "**"
                    initializer = enumvalue.find("initializer")
                    if initializer is not None:
                        p += f" {self.parser.paras_as_str(initializer, plain=plain)}"
                    ret.append(p)
            return ret

        def has(self) -> bool:
            return self.xmlxml.find("enumvalue") is not None if self.kindkind.is_enum() else False

    class Initializer:
        def __init__(self, xml: Element, parser: XmlParser, kind: Kind):
            self.xmlxml = xml
            self.parserparser = parser
            self.kindkind = kind

        def md(self, plain: bool = False) -> str:
            initializer = self.xmlxml.find("initializer")
            if initializer is not None:
                initializer_str = self.parserparser.paras_as_str(initializer, plain=plain).lstrip(" =")
                if "\n" in initializer_str:
                    return "`/* multi line expression */`"
                return f"`{initializer_str}`"
            else:
                return ""

        def plain(self) -> str:
            return self.mdmd(plain=True)

        def has(self) -> bool:
            return self.xmlxml.find("initializer") is not None

    class Definition:
        def __init__(self, xml: Element, parser: XmlParser, kind: Kind):
            self.xmlxml = xml
            self.parserparser = parser
            self.kindkind = kind

        def md(self, plain: bool = False) -> str:
            return self.plainplain()

        def plain(self) -> str:
            definition = self.xmlxml.find("definition")
            if definition is not None and definition.text:
                return f"{definition.text};"
            else:
                return ""

        def has(self) -> bool:
            return self.xmlxml.find("definition") is not None

    class Programlisting:
        def __init__(self, xml: Element, parser: XmlParser, kind: Kind):
            self.xmlxml = xml
            self.parserparser = parser
            self.kindkind = kind

        def md(self, plain: bool = False) -> str:
            programlisting = self.xmlxml.find("programlisting")
            if programlisting is None:
                return ""

            return self.parserparser.programlisting_as_str(programlisting)

        def has(self) -> bool:
            return self.xmlxml.find("programlisting") is not None