"""
domonic.dom
===========
The core DOM implementation for domonic.
This module provides the document tree, node and element types, collections,
range and selection helpers, geometry interfaces, mutation and layout
observers, and the document-facing APIs that the rest of the package builds on.
It is intended to feel like a practical Python surface for the DOM and related
web-platform concepts rather than a small HTML helper tree.
"""
from __future__ import annotations
import copy
import os
import re
import time
from email.utils import formatdate
from typing import Any, Callable, ClassVar, Iterable, Iterator
from domonic.events import Event, EventTarget, MouseEvent
from domonic.geom.vec3 import vec3
from domonic.style import CSSStyleDeclaration as Style
from domonic.style import StyleSheetList
from domonic.webapi.console import Console
from domonic.webapi.url import URL
from domonic.webapi.xpath import (XPathEvaluator, XPathException,
XPathExpression, XPathResult)
# from xml.dom.pulldom import END_ELEMENT
[docs]
class DOMConfig:
"""Global rendering and behaviour flags for domonic's DOM.
``DOMConfig`` controls how trees are rendered and how a few optional
behaviours are interpreted across the library, such as auto-escaping text
content and optional closing-tag handling.
"""
GLOBAL_AUTOESCAPE: bool = False # Default is False
RENDER_OPTIONAL_CLOSING_TAGS: bool = True # Default is True
RENDER_OPTIONAL_CLOSING_SLASH: bool = True # on emtpy nodes should the last slash be rendered
SPACE_BEFORE_OPTIONAL_CLOSING_SLASH: bool = (
False # on emtpy nodes should there be a space before the closing slash?
)
HTMX_ENABLED: bool = False # Default is false
# NO_REPR: bool = True # objects always render?
ATTRIBUTE_QUOTES: bool | str | None = '"' # i.e. <tag="">
def _get_custom_element_registry():
try:
from domonic.window import window as domonic_window
except Exception:
return None
return getattr(domonic_window, "customElements", None)
def _iter_dom_nodes(node):
if not isinstance(node, Node):
return
yield node
for child in getattr(node, "childNodes", []):
if isinstance(child, Node):
yield from _iter_dom_nodes(child)
def _node_is_connected(node: "Node") -> bool:
try:
return isinstance(node.rootNode, Document)
except Exception:
return False
def _notify_attribute_changed(element: "Element", attribute: str, old_value: Any, new_value: Any) -> None:
callback = getattr(element, "attributeChangedCallback", None)
if not callable(callback) or old_value == new_value:
return
observed = getattr(element.__class__, "observedAttributes", ())
if observed is None:
observed = ()
normalized = attribute[1:] if attribute.startswith("_") else attribute
if normalized in tuple(observed):
callback(normalized, old_value, new_value)
def _run_connected_callback(element: "Element") -> None:
callback = getattr(element, "connectedCallback", None)
if callable(callback) and not getattr(element, "_custom_element_connected", False):
element._custom_element_connected = True
callback()
def _run_disconnected_callback(element: "Element") -> None:
callback = getattr(element, "disconnectedCallback", None)
if callable(callback) and getattr(element, "_custom_element_connected", False):
element._custom_element_connected = False
callback()
def _run_adopted_callback(element: "Element", old_document: "Document | None", new_document: "Document | None") -> None:
callback = getattr(element, "adoptedCallback", None)
if callable(callback) and old_document is not new_document:
callback(old_document, new_document)
def _adopt_tree(node: "Node", old_document: "Document | None", new_document: "Document | None") -> None:
for current in _iter_dom_nodes(node):
current._ownerDocument = new_document
if isinstance(current, Element):
_run_adopted_callback(current, old_document, new_document)
def _upgrade_custom_element_instance(element: "Element") -> "Element":
registry = _get_custom_element_registry()
if registry is None:
return element
return registry._upgrade_element(element)
def _connect_tree(node: "Node") -> None:
for current in _iter_dom_nodes(node):
current._ownerDocument = current.rootNode if isinstance(current.rootNode, Document) else getattr(current, "_ownerDocument", None)
current.isConnected = _node_is_connected(current)
if isinstance(current, Element):
_upgrade_custom_element_instance(current)
if current.isConnected:
_run_connected_callback(current)
def _disconnect_tree(node: "Node") -> None:
for current in _iter_dom_nodes(node):
current.isConnected = False
if isinstance(current, Element):
_run_disconnected_callback(current)
def _assigned_slot_for_node(node: "Node") -> "HTMLSlotElement | None":
parent = getattr(node, "parentNode", None)
if not isinstance(parent, Element):
return None
shadow_root = getattr(parent, "shadowRoot", None)
if not isinstance(shadow_root, ShadowRoot):
return None
slot_name = ""
if isinstance(node, Element):
slot_name = node.getAttribute("slot") or ""
for child in shadow_root.childNodes:
if isinstance(child, HTMLSlotElement):
if (child.getAttribute("name") or "") == slot_name:
return child
if slot_name == "":
for child in shadow_root.childNodes:
if isinstance(child, HTMLSlotElement) and not child.getAttribute("name"):
return child
return None
def _notify_slot_change(target: "Node") -> None:
slots: list[HTMLSlotElement] = []
if isinstance(target, ShadowRoot):
slots = [child for child in target.childNodes if isinstance(child, HTMLSlotElement)]
elif isinstance(target, Element) and isinstance(getattr(target, "shadowRoot", None), ShadowRoot):
slots = [child for child in target.shadowRoot.childNodes if isinstance(child, HTMLSlotElement)]
for slot in slots:
slot.dispatchEvent(Event("slotchange"))
def _iter_ancestors_inclusive(node: "Node | None") -> Iterator["Node"]:
current = node
while isinstance(current, Node):
yield current
current = getattr(current, "parentNode", None)
def _normalize_mutation_observer_options(options: dict[str, Any]) -> dict[str, Any]:
normalized = {
"subtree": bool(options.get("subtree", False)),
"childList": bool(options.get("childList", False)),
"attributes": bool(options.get("attributes", False)),
"attributeFilter": options.get("attributeFilter"),
"attributeOldValue": bool(options.get("attributeOldValue", False)),
"characterData": bool(options.get("characterData", False)),
"characterDataOldValue": bool(options.get("characterDataOldValue", False)),
}
if normalized["attributeFilter"] is not None or normalized["attributeOldValue"]:
normalized["attributes"] = True
if normalized["characterDataOldValue"]:
normalized["characterData"] = True
if normalized["attributeFilter"] is not None:
normalized["attributeFilter"] = tuple(
attr[1:] if isinstance(attr, str) and attr.startswith("_") else attr
for attr in normalized["attributeFilter"]
)
if not any(
(
normalized["childList"],
normalized["attributes"],
normalized["characterData"],
)
):
raise TypeError("MutationObserver options must enable childList, attributes, or characterData")
return normalized
def _queue_mutation_record(
record_type: str,
target: "Node",
*,
added_nodes: Iterable["Node"] | None = None,
removed_nodes: Iterable["Node"] | None = None,
previous_sibling: "Node | None" = None,
next_sibling: "Node | None" = None,
attribute_name: str | None = None,
attribute_namespace: str | None = None,
old_value: str | None = None,
) -> None:
try:
observers = list(MutationObserver._all_observers)
except NameError:
return
if not observers:
return
record = MutationRecord(
record_type,
target,
addedNodes=added_nodes or (),
removedNodes=removed_nodes or (),
previousSibling=previous_sibling,
nextSibling=next_sibling,
attributeName=attribute_name,
attributeNamespace=attribute_namespace,
oldValue=old_value,
)
pending: list[MutationObserver] = []
for observer in observers:
if observer._enqueue_if_observing(record):
pending.append(observer)
for observer in pending:
observer._flush()
_process_observer_notifications(target)
_observer_processing: bool = False
def _intersect_rects(first: DOMRectReadOnly, second: DOMRectReadOnly) -> DOMRect:
left = max(first.left, second.left)
top = max(first.top, second.top)
right = min(first.right, second.right)
bottom = min(first.bottom, second.bottom)
if right <= left or bottom <= top:
return DOMRect(left, top, 0, 0)
return DOMRect(left, top, right - left, bottom - top)
def _default_intersection_root_rect(target: "Element", target_rect: DOMRectReadOnly) -> DOMRectReadOnly:
doc = target.ownerDocument if isinstance(target.ownerDocument, Document) else None
root = None
if doc is not None:
root = getattr(doc, "documentElement", None) or getattr(doc, "body", None)
if isinstance(root, Element) and root is not target:
return root.getBoundingClientRect()
return DOMRectReadOnly.fromRect(target_rect)
def _process_observer_notifications(target: "Node | None" = None, target_rect: DOMRectReadOnly | None = None) -> None:
global _observer_processing
if _observer_processing:
return
_observer_processing = True
try:
try:
resize_observers = list(ResizeObserver._all_observers)
except NameError:
resize_observers = []
for observer in resize_observers:
observer._process(target, target_rect)
try:
intersection_observers = list(IntersectionObserver._all_observers)
except NameError:
intersection_observers = []
for observer in intersection_observers:
observer._process(target, target_rect)
finally:
_observer_processing = False
def _form_owner(control: "Element") -> "HTMLFormElement | None":
owner_document = control.ownerDocument if isinstance(control.ownerDocument, Document) else None
form_id = control.getAttribute("form") if isinstance(control, Element) else None
if form_id and owner_document is not None:
form = owner_document.getElementById(form_id)
if isinstance(form, HTMLFormElement):
return form
for ancestor in _iter_ancestors_inclusive(getattr(control, "parentNode", None)):
if isinstance(ancestor, HTMLFormElement):
return ancestor
return None
def _dispatch_value_change_events(control: "Element") -> None:
from domonic.events import Event, InputEvent
control.dispatchEvent(InputEvent("input", {"bubbles": True, "cancelable": False}))
control.dispatchEvent(Event("change", {"bubbles": True, "cancelable": False}))
def _is_control_valid(control: "Element") -> bool:
if not isinstance(control, Element) or control.hasAttribute("disabled"):
return True
if control.hasAttribute("required"):
tag_name = getattr(control, "tagName", getattr(control, "name", "")).lower()
if tag_name == "input":
input_type = (control.getAttribute("type") or "text").lower()
if input_type in {"checkbox", "radio"}:
return control.checked
return control.value != ""
if tag_name == "textarea":
return control.value != ""
if tag_name == "select":
return control.value != ""
return True
[docs]
class Node(EventTarget):
"""An abstract base class upon which many other DOM API objects are based"""
ELEMENT_NODE: int = 1
TEXT_NODE: int = 3
CDATA_SECTION_NODE: int = 4
PROCESSING_INSTRUCTION_NODE: int = 7
COMMENT_NODE: int = 8
DOCUMENT_NODE: int = 9
DOCUMENT_TYPE_NODE: int = 10
DOCUMENT_FRAGMENT_NODE: int = 11
DOCUMENT_POSITION_DISCONNECTED: int = 1
DOCUMENT_POSITION_PRECEDING: int = 2
DOCUMENT_POSITION_FOLLOWING: int = 4
DOCUMENT_POSITION_CONTAINS: int = 8
DOCUMENT_POSITION_CONTAINED_BY: int = 16
DOCUMENT_POSITION_IMPLEMENTATION_SPECIFIC: int = 32
# The following constants have been deprecated and should not be used anymore.
ATTRIBUTE_NODE: int = 2
ENTITY_REFERENCE_NODE: int = 5
ENTITY_NODE: int = 6
NOTATION_NODE: int = 12
__isempty: bool = False # tells us if the node is empty i.e. has no content aka 'self closing'. in html that would be: area, base, br, col, embed, hr, img, input, link, meta, param, source, track, True
__context: ClassVar[list["Node"] | None] = None # private. tags will append to last item in context on creation.
# __slots__ = ['____attributes__',
# '__content',
# 'name',
# '__rootNode',
# 'parentNode',
# 'baseURI',
# 'isConnected',
# 'namespaceURI',
# 'outerText',
# 'prefix']
def __init__(self, *args, **kwargs) -> None:
self.args = args
self.kwargs = kwargs
if getattr(self, "name", None) is None:
self.name = ""
# if user doesn't put underscore (dont advertise this as still has issues.)
new_kwargs = {}
for k, v in kwargs.items():
if k[0] != "_":
new_kwargs[f"_{k}"] = v
else:
new_kwargs[k] = v
self.kwargs = new_kwargs
self.baseURI: str = "" # TODO - if ownerdocument has a basetag, use that
self.isConnected: bool = True
self.namespaceURI: str = "http://www.w3.org/1999/xhtml"
self.outerText: str = None
self._ownerDocument = None
self.parentNode = None
self.prefix = None # 🗑️
# self.baseURIObject = None # ?
# self.nodePrincipal = None
self._update_parents()
# attempt to set init namespaceURI based on the tag name
try:
n = self.rootNode
nm = n.tagName
# print(n)
if nm == "html":
self.namespaceURI = "http://www.w3.org/1999/xhtml"
elif nm == "svg":
self.namespaceURI = "http://www.w3.org/2000/svg"
elif nm == "xhtml":
self.namespaceURI = "http://www.w3.org/1999/xhtml"
elif nm == "xml":
self.namespaceURI = "http://www.w3.org/XML/1998/namespace"
elif nm == "xlink":
self.namespaceURI = "http://www.w3.org/1999/xlink"
elif nm == "math":
self.namespaceURI = "http://www.w3.org/1998/Math/MathML"
# elif nm == "rdf":
# self.namespaceURI = "http://www.w3.org/1999/02/22-rdf-syntax-ns#"
# elif nm == "rdfs":
# self.namespaceURI = "http://www.w3.org/2000/01/rdf-schema#"
# elif nm == "owl":
# self.namespaceURI = "http://www.w3.org/2002/07/owl#"
# elif nm == "dc":
# self.namespaceURI = "http://purl.org/dc/elements/1.1/"
# elif nm == "dcterms":
# self.namespaceURI = "http://purl.org/dc/terms/"
# elif nm == "foaf":
# self.namespaceURI = "http://xmlns.com/foaf/0.1/"
# elif nm == "cc":
# self.namespaceURI = "http://web.resource.org/cc/"
# elif nm == "gr":
# self.namespaceURI = "http://purl.org/goodrelations/v1#"
# elif nm == "sioc":
# self.namespaceURI = "http://rdfs.org/sioc/ns#"
# elif nm == "doap":
# self.namespaceURI = "http://usefulinc.com/ns/doap#"
# elif nm == "vcard":
# self.namespaceURI = "http://www.w3.org/2006/vcard/ns#"
# elif nm == "schema":
# self.namespaceURI = "http://schema.org/"
# elif nm == "og":
# self.namespaceURI = "http://ogp.me/ns#"
# elif nm == "geo":
# self.namespaceURI = "http://www.w3.org/2003/01/geo/wgs84_pos#"
# elif nm == "rev":
# self.namespaceURI = "http://purl.org/stuff/rev#"
# elif nm == "sioc":
# self.namespaceURI = "http://rdfs.org/sioc/ns#"
# elif nm == "skos":
# self.namespaceURI = "http://www.w3.org/2004/02/skos/core#" # TODO - test
# elif nm == "wot":
# self.namespaceURI = "http://xmlns.com/wot/0.1/"
# elif nm == "wgs84_pos":
# self.namespaceURI = "http://www.w3.org/2003/01/geo/wgs84_pos#"
# elif nm == "xhv":
# self.namespaceURI = "http://www.w3.org/1999/xhtml/vocab#"
except Exception as e:
pass
# this is for using 'with'
if Node.__context is not None:
Node.__context[len(Node.__context) - 1] += self
super().__init__(*args, **kwargs)
@property
def content(self):
# return ''.join([each.__str__() for each in self.args])
# if any child are lists by mistake, loop and call __str__ on each first
cnt = self.args
for i, arg in enumerate(cnt):
if isinstance(arg, list):
cnt = list(cnt)
cnt[i] = "".join([each.__str__() for each in arg])
cnt = tuple(cnt)
if DOMConfig.GLOBAL_AUTOESCAPE:
import html as fix
cnt = list(cnt)
for each, child in enumerate(cnt):
if isinstance(child, str) or isinstance(child, Text):
child = fix.escape(str(child))
cnt[each] = child
cnt = tuple(cnt)
return "".join([each.__str__() for each in cnt])
# else:
return "".join([each.__str__() for each in cnt])
@content.setter
def content(self, ignore):
self.__content = "".join([each.__str__() for each in self.args])
return
@property
def __attributes__(self):
def format_attr(key, value):
if value is True:
value = "true"
if value is False:
value = "false"
key = key.split("_", 1)[1]
QM = DOMConfig.ATTRIBUTE_QUOTES
if DOMConfig.ATTRIBUTE_QUOTES is False or DOMConfig.ATTRIBUTE_QUOTES == "":
QM = ""
elif DOMConfig.ATTRIBUTE_QUOTES is True or DOMConfig.ATTRIBUTE_QUOTES is None:
QM = '"'
# note - consider making this an attributes handler for any custom attributes
# so on config user can add a handler function for the attribute
if DOMConfig.HTMX_ENABLED:
# if htmx is enabld
htmx_attributes = [
"boost",
"confirm",
"delete",
"disable",
"disinherit",
"encoding",
"ext",
"get",
"headers",
"history_elt",
"include",
"indicator",
"params",
"patch",
"post",
"preserve",
"prompt",
"push_url",
"put",
"request",
"select",
"sse",
"swap",
"swap_oob",
"sync",
"target",
"trigger",
"vals",
"vars",
"ws",
]
if key in htmx_attributes:
return f""" data-hx-{key}={QM if DOMConfig.ATTRIBUTE_QUOTES is not None else QM if type(value) == str else ''}{value}{QM if DOMConfig.ATTRIBUTE_QUOTES is not None else QM if type(value) == str else ''}"""
# lets us have boolean attributes
if key in [
"async",
"checked",
"autofocus",
"disabled",
"formnovalidate",
"hidden",
"multiple",
"novalidate",
"readonly",
"required",
"selected",
"open",
"contenteditable",
"reversed",
"download",
"draggable",
"spellcheck",
"translate",
"autoplay", # Added
"controls", # Added
"loop", # Added
"muted", # Added
"default", # Added
"allowfullscreen",# Added
"playsinline", # Added
"value", # Added
"defer", # Added
# "compact", # Added
# "ismap", # Added
# "sandbox", # Added
# "seamless", # Added
# "selected", # Added
# "sortable", # Added
# "truespeed", # Added
# "typemustmatch", # Added
# "visible", # Added
# "wrap", # Added
# "novalidate", # Added
# "open", # Added
# "readonly", # Added
# "required", # Added
]:
if value == "" or value == key:
return f""" {key}"""
return f""" {key}={QM if DOMConfig.ATTRIBUTE_QUOTES is not None else QM if type(value) == str else ''}{value}{QM if DOMConfig.ATTRIBUTE_QUOTES is not None else QM if type(value) == str else ''}"""
try:
return "".join([format_attr(key, value) for key, value in self.kwargs.items()])
except IndexError as e:
from domonic.html import TemplateError
raise TemplateError(e)
# except Exception as e:
# print(e)
@__attributes__.setter
def __attributes__(self, ignore):
try:
QM = DOMConfig.ATTRIBUTE_QUOTES
if DOMConfig.ATTRIBUTE_QUOTES is False or DOMConfig.ATTRIBUTE_QUOTES == "":
QM = ""
elif DOMConfig.ATTRIBUTE_QUOTES is True or DOMConfig.ATTRIBUTE_QUOTES is None:
QM = '"'
self.__attributes = "".join(
[
f""" {key.split('_', 1)[1]}={QM if DOMConfig.ATTRIBUTE_QUOTES is not None else QM if type(value) == str else ''}{value}{QM if DOMConfig.ATTRIBUTE_QUOTES is not None else QM if type(value) == str else ''}"""
for key, value in self.kwargs.items()
]
)
except IndexError as e:
from domonic.html import TemplateError
raise TemplateError(e)
# except Exception as e:
# print(e)
def __str__(self):
if not DOMConfig.RENDER_OPTIONAL_CLOSING_TAGS:
if self.name in [
"html",
"head",
"body",
"p",
"dt",
"dd",
"li",
"option",
"thead",
"th",
"tbody",
"tr",
"td",
"tfoot",
"colgroup",
]:
return f"<{self.name}{self.__attributes__}>{self.content}"
return f"<{self.name}{self.__attributes__}>{self.content}</{self.name}>"
def __mul__(self, other):
"""
requires you to render yourself i.e.
cells = cell()*10
print(''.join([str(c) for c in cells]))
"""
reproducer = []
for i in range(other):
reproducer.append(copy.deepcopy(self))
return reproducer
def __rmul__(self, other):
"""
requires you to render yourself i.e.
cells = cell()*10
print(''.join([str(c) for c in cells]))
"""
reproducer = []
for i in range(other):
reproducer.append(copy.deepcopy(self))
return reproducer
def __truediv__(self, other):
"""use to render clones without having to parse commas yourself"""
reproducer = []
for i in range(other):
reproducer.append(str(self))
return "".join(reproducer)
def __rtruediv__(self, other):
"""use to render clones without having to parse commas yourself"""
reproducer = []
for i in range(other):
reproducer.append(str(self))
return "".join(reproducer)
def __div__(self, other):
"""
useful for prototyping as renders. to retain objects use multiply
"""
reproducer = []
for i in range(other):
reproducer.append(str(self))
return "".join(reproducer)
def __rdiv__(self, other):
"""
useful for prototyping as renders. to retain objects use multiply
"""
reproducer = []
for i in range(other):
reproducer.append(str(self))
return "".join(reproducer)
def __or__(self, other):
"""return self unless other is something"""
if other is not False:
return other
return self
def __iadd__(self, item):
"""adds an item to the nodes of children. can also pass a list and it will unpack them"""
if isinstance(item, (list, tuple)): # TODO - Documentfragment?
for i in item:
self.args = self.args + (i,)
return self
self.args = self.args + (item,)
return self
def __isub__(self, item):
"""removes an item from the list of children"""
replace_args = list(self.args)
replace_args.remove(item)
self.args = tuple(replace_args)
return self
def __getitem__(self, index):
if isinstance(index, int):
return self.args[index]
# elif isinstance(index, str):
# if index.startswith('_'):
# return self.kwargs[index]
# else:
# return getattr(self, index)
# super(Node, self).__getitem__(index)
if isinstance(index, str):
# call props on self
# print('erk!')
try:
# return Node.__dict__[index]
return getattr(self, index)
except Exception as e:
print(e)
# return None
# return super(Node, self).__getitem__(index)
def __rshift__(self, item):
try:
for key in item.keys():
self.kwargs[key] = item[key]
return self
except Exception as e:
print(e)
raise ValueError
# def __add__(self, item):
# try:
# self.args = self.args + (item,)
# return self
# except Exception as e:
# print(e)
# raise ValueError
# def __sub__(self, item):
# try:
# self.args = self.args - (item,)
# return self
# except Exception as e:
# print(e)
# raise ValueError
# def render()
def __getattr__(self, attr):
"""
allows dot notation for reading attributes
*credit to the peeps on discord/python for this one*
"""
try:
kwargs = super().__getattribute__("kwargs")
except AttributeError:
kwargs = {}
if attr in kwargs:
return kwargs[attr]
retry = "_" + attr
if retry in kwargs:
return kwargs[retry]
retry = attr[1 : len(attr)]
if retry in kwargs:
return kwargs[retry]
# TODO - think of solution for other MIA attributes as when it would fail silently
# it was a nightmare. But having to catch the raised errors may also be sluggish
# maybe specific tags can override this method and provide default values when not present?
if self.__class__.__name__ == "a" and attr == "href":
print(" Warning: No 'href' attribute was defined for this 'a' tag.")
return ""
try:
# return getattr(super(), attr)
# return getattr(self, attr)
# return getattr(Node, attr) # means overrideing for style etc in element?
return getattr(self.__class__, attr) # means overrideing for style etc in element?
# return getattr(Element, attr)
except AttributeError as e:
# print(e) # TODO - careful. better on for debugging.
raise e # ("attribute does not exist:", attr)
raise AttributeError
def __pyml__(self):
"""[returns a representation of the object as a pyml string]"""
# from domonic.dom import Text
params = ""
for key, value in self.kwargs.items():
if "-" in key:
params += f'**\u007b"{key}":{value}\u007d,'
else:
params += f'{key}="{value}", '
# TODO - will need to loop args and call __pyml__ on each one
for arg in self.args:
try:
if isinstance(arg, Text):
params += '"' + str(arg) + '"' + ", "
else:
params += f"{arg.__pyml__()}, "
except Exception as e:
params += str(arg) + ", "
# TODO - if self is document do dentage
return f"{self.name}({params[:-2]})"
# return f"{self.name}({params})"
# return f"{self.name}({args}, {params})"
# return f"<{self.name}{self.__attributes__}>{self.content}</{self.name}>"
def __repr__(self) -> str:
name = self.name or self.__class__.__name__
return f"<{name}{self.__attributes__}>"
def _repr_html_(self) -> str:
return str(self)
def __setitem__(self, key, value):
try:
self.kwargs[key] = value
return self
except Exception as e:
print(e)
raise ValueError
def __enter__(self):
if Node.__context is None:
Node.__context = []
Node.__context.append(self)
return self
def __exit__(self, type, value, traceback, *args, **kwargs):
Node.__context.pop()
if len(Node.__context) == 0:
Node.__context = None
return self
# def __dir__(self):
# return self.__dict__.keys()
# TODO - these are hard and wil need tests
# def __setattr__(self, attr, value):
# def __delattr__(self, attr):
# def __next__(self):
# def __iter__(self):
def __iter__(self):
return iter(self.args)
def __format__(self, format_spec):
# return super().__format__(format_spec)
# get node depth by counting parents
# TODO - this is a hack to get the depth of the node
n = self
depth = 0
while n is not None:
# print(type(n), type(n.parentNode))
n = n.parentNode
depth += 1
depth -= 1
# print(f"depth: {depth}")
# dent = ' ' * depth
dent = "\t" * depth
# loop the children and call __format__ on each one
# content = ""
# for child in self.childNodes:
# content += child.__format__(format_spec)
self._update_parents()
if DOMConfig.GLOBAL_AUTOESCAPE: # TODO - unit tests
import html as fix
self.args = list(self.args)
for each, child in enumerate(self.args):
if isinstance(child, str) or isinstance(child, Text):
child = fix.escape(str(child))
self.args[each] = child
self.args = tuple(self.args)
content = "".join([each.__format__(format_spec) for each in self.args])
# from concurrent.futures import ThreadPoolExecutor
# content = ''
# with ThreadPoolExecutor(10) as executor:
# for result in executor.map(lambda x: x.__format__(format_spec), self.args):
# content += result
wrap = False
if len(self.args) == 1:
if not isinstance(self.args[0], Element):
wrap = True
dtype = ""
if isinstance(self, Document):
# dtype = "<!DOCTYPE html>"
dtype = self.doctype
# if self is a closed_tag, return the content
from domonic.html import closed_tag
if isinstance(self, closed_tag):
return f"\n{dent}<{self.name}{self.__attributes__} />"
# in html5 the following tags are optional closing tags
# html, head, body, p, dt, dd, li, option, thead, th, tbody, tr, td, tfoot, colgroup
size = len(str(content))
if DOMConfig.RENDER_OPTIONAL_CLOSING_TAGS:
if size < 150 and wrap:
return f"\n{dent}<{self.name}{self.__attributes__}>{content}</{self.name}>"
else:
return f"{dtype}\n{dent}<{self.name}{self.__attributes__}>{content}\n{dent}</{self.name}>"
else:
if self.name in [
"html",
"head",
"body",
"p",
"dt",
"dd",
"li",
"option",
"thead",
"th",
"tbody",
"tr",
"td",
"tfoot",
"colgroup",
]:
if size < 150 and wrap:
return f"\n{dent}<{self.name}{self.__attributes__}>{content}"
else:
return f"{dtype}\n{dent}<{self.name}{self.__attributes__}>{content}\n"
else:
if size < 150 and wrap:
return f"\n{dent}<{self.name}{self.__attributes__}>{content}</{self.name}>"
else:
return f"{dtype}\n{dent}<{self.name}{self.__attributes__}>{content}\n{dent}</{self.name}>"
# def __call__(self, *args, **kwargs):
# """
# allows for calling the object as a function
# """
# print('calling a tag')
# print(args)
# print(kwargs)
# print(self.name)
def __setattr__(self, name: str, value: Any) -> None:
try:
if name == "args":
super().__setattr__(name, value)
self._update_parents()
return
except Exception as e:
print(e)
# pass
super().__setattr__(name, value)
# def __getattr__(self, name):
# # print(name)
# try:
# if name == "args":
# return super(Node, self).__getattr__(name)
# except Exception as e:
# print(e)
# return super(Node, self).__getattr__(name)
# def __getattribute__(self, name):
# print('how are you doing today', name)
# try:
# if name == "args":
# return super(Node, self).__getattribute__(name)
# except Exception as e:
# print(e)
# check if its a property on the class
# if name in self.__dict__:
# return super(Node, self).__getattribute__(name)
# return super(Node, self).__getattribute__(name)
# return self.__dict__[item]
# def __getattr__(self, attrName):
# if name not in self.__dict__:
# value = self.fetchAttr(name) # computes the value
# self.__dict__[name] = value
# return self.__dict__[name]
# TODO - html.tag class currently has the required method as its called there.
# def __getitem__(self, item):
# print('GET ITEM CALLED')
# if isinstance(item, int):
# return self.childNodes[item]
# if isinstance(item, str):
# # call props on self
# print('sup!')
# try:
# return self.__dict__[item]
# except Exception as e:
# print(e)
# # return None
# return super(Node, self).__getitem__(item)
def _update_parents(self):
"""private. - TODO < check these docstrings don't export in docs
loops all children and sets self as parent.
cant do as decorator for now as that seems to breaks potential for json serialisation (see Style)
so will have to call manually whenever self.args are ammended.
"""
try:
# print(self.args)
for el in self.args:
# if(type(el) not in [str, list, dict, int, float, tuple, object, set]):
if isinstance(el, (Element, Node)):
el.parentNode = self
el._update_parents()
except Exception as e:
print("unable to update parent", e)
def _iterate(self, element, callback) -> None:
"""private. - TODO < check these docstrings don't export in docs
loops all children and sets self as parent.
cant do as decorator for now as that seems to breaks potential for json serialisation (see Style)
so will have to call manually whenever self.args are ammended.
"""
callback(element) # TODO - this can block on failed attributes
elements = []
if isinstance(element, Node):
elements = element.args
elif isinstance(element, list):
elements = element
try:
for el in elements:
if type(el) not in [str, list, dict, int, float, tuple, object, set]:
# callback(el)
el._iterate(el, callback)
elif isinstance(el, list): # if someone is incorrectly using a list as a child
for e in el:
if type(e) not in (str, list, dict, int, float, tuple, object, set):
e._iterate(e, callback)
except Exception:
return
def __bool__(self) -> bool:
# Nodes should be truthy by existence, not by child count.
return True
def __len__(self) -> int:
return len(self.args)
def __contains__(self, item: Any) -> bool:
if item in self.args:
return True
return self.contains(item) if isinstance(item, Node) else False
@property
def assignedSlot(self):
return _assigned_slot_for_node(self)
[docs]
def appendChild(self, aChild: "Node") -> "Node":
"""
Adds a child to the current element.
If item is a DocumentFragment, all its children are added.
Args:
item (Node): The Node to add.
"""
if isinstance(aChild, DocumentFragment):
items = aChild.args
previous_sibling = self.args[-1] if len(self.args) else None
self.args = self.args + items
for item in items:
if isinstance(item, Node):
old_document = item.ownerDocument if isinstance(item.ownerDocument, Document) else None
item.parentNode = self
_adopt_tree(item, old_document, self.ownerDocument if isinstance(self.ownerDocument, Document) else None)
_connect_tree(item)
added_nodes = [item for item in items if isinstance(item, Node)]
if added_nodes:
_queue_mutation_record("childList", self, added_nodes=added_nodes, previous_sibling=previous_sibling)
_notify_slot_change(self)
return DocumentFragment()
else:
previous_sibling = self.args[-1] if len(self.args) else None
self.args = self.args + (aChild,)
if isinstance(aChild, Node):
old_document = aChild.ownerDocument if isinstance(aChild.ownerDocument, Document) else None
aChild.parentNode = self
_adopt_tree(aChild, old_document, self.ownerDocument if isinstance(self.ownerDocument, Document) else None)
_connect_tree(aChild)
_queue_mutation_record("childList", self, added_nodes=(aChild,), previous_sibling=previous_sibling)
_notify_slot_change(self)
# return aChild # causes max recursion when called chained? then don't chain?
return aChild
@property
def childElementCount(self) -> int:
"""Returns the number of child elements an element has"""
return len(self.args)
@property
def childNodes(self) -> "NodeList":
"""Returns a live NodeList containing all the children of this node"""
# return list(self.args)
return NodeList(self.args)
@property
def children(self) -> list[Node]:
"""Returns a collection of an element's child element (excluding text and comment nodes)"""
newlist: list = []
for each in self.args:
if type(each) != str:
newlist.append(each)
return newlist
[docs]
def compareDocumentPosition(self, otherElement: "Node") -> int:
"""
An integer value representing otherNode's position relative to node as a bitmask combining the following constant properties of Node:
https://stackoverflow.com/questions/8334286/cross-browser-compare-document-position
"""
thisNode = self
other = otherElement
# if isinstance(other, str):
# other = Text(other)
# if isinstance(thisNode, str):
# thisNode = Text(thisNode)
def recursivelyWalk(nodes, cb):
for node in nodes:
if isinstance(node, str):
node = Text(node)
# continue
ret = cb(node)
if ret:
return ret
if node.childNodes and node.childNodes.length > 0:
ret = recursivelyWalk(node.childNodes, cb)
if ret:
return ret
def testNodeForComparePosition(node, other):
if node is other:
return True
def identifyWhichIsFirst(node):
if node == other:
return "other"
elif node == reference:
return "reference"
reference = thisNode
referenceTop = thisNode
otherTop = other
if self == other:
return 0
while referenceTop.parentNode is not None:
referenceTop = referenceTop.parentNode
while otherTop.parentNode is not None:
otherTop = otherTop.parentNode
# print(referenceTop, otherTop)
if referenceTop != otherTop:
return Node.DOCUMENT_POSITION_DISCONNECTED
children = reference.childNodes
ret = recursivelyWalk(children, lambda p: testNodeForComparePosition(other, p))
if ret:
return Node.DOCUMENT_POSITION_CONTAINED_BY # + Node.DOCUMENT_POSITION_FOLLOWING
children = other.childNodes
ret = recursivelyWalk(children, lambda p: testNodeForComparePosition(reference, p))
if ret:
return Node.DOCUMENT_POSITION_CONTAINS # + Node.DOCUMENT_POSITION_PRECEDING
ret = recursivelyWalk([referenceTop], identifyWhichIsFirst)
if ret == "other":
return Node.DOCUMENT_POSITION_PRECEDING
else:
return Node.DOCUMENT_POSITION_FOLLOWING
[docs]
def contains(self, node: "Node") -> bool:
"""Check whether a node is a descendant of a given node"""
# this will go crunch on big stuff... need to consider best way
for each in self.args:
if each == node:
return True
try:
if each.contains(node):
return True
except Exception:
pass # TODO - dont iterate strings
return False
@property
def firstChild(self) -> Node | None:
"""Returns the first child node of an element"""
try:
return self.args[0] # TODO - check if this means includes content
except Exception:
return None
[docs]
def hasChildNodes(self) -> bool:
"""Returns true if an element has any child nodes, otherwise false"""
return len(self.args) > 0
@property
def lastChild(self) -> Node | None:
"""Returns the last child node of an element"""
try:
return self.args[len(self.args) - 1]
except Exception:
return None
@property
def localName(self) -> str | None:
try:
return self.tagName
except Exception:
return None
@property
def nodeName(self) -> str | None:
"""Returns the name of a node"""
# TODO - not sure what's better this or overriding on every element
# if isinstance(self, Text):
# return '#text'
# if isinstance(self, Comment):
# return '#comment'
# elif isinstance(self, DocumentType):
# return '#doctype'
if isinstance(self, Document): # NOTE - having this one on breaks parser. as it expects 'html'?
return "#document"
if isinstance(self, CDATASection):
return "#cdata-section"
elif isinstance(self, DocumentFragment):
return "#document-fragment"
elif isinstance(self, Attr):
return self.name
elif isinstance(self, ProcessingInstruction):
return self.target
elif isinstance(self, DocumentType):
return self.name
# print(type(self))
if isinstance(self, Element):
return self.tagName # .upper()
else:
try:
return self.tagName
except Exception:
return None
nodeType: int = ELEMENT_NODE
@property
def nodeValue(self) -> str | None:
"""Sets or returns the value of a node"""
outp = ""
for each in self.args:
if type(each) is str:
outp = outp + each
else:
val = each.nodeValue
if val is not None:
outp = outp + val
else:
return None
if outp == "":
outp = None
return outp
@nodeValue.setter
def nodeValue(self, content: Any):
"""Sets or returns the value of a node"""
old_value = self.nodeValue
self.args = (content,)
if isinstance(self, CharacterData):
_queue_mutation_record("characterData", self, old_value=old_value)
return content
@property
def ownerDocument(self) -> "Node | None":
"""Returns the root element (document object) for an element"""
root = self.rootNode
if isinstance(root, Document):
return root
return getattr(self, "_ownerDocument", None)
@ownerDocument.setter
def ownerDocument(self, newOwner: Node | None): #: Element):
"""Sets the root element (document object) for an element"""
if newOwner is None:
return
self._ownerDocument = newOwner if isinstance(newOwner, Document) else getattr(newOwner, "ownerDocument", None)
@property
def rootNode(self) -> "Node":
"""[read-only property returns a Node object representing the topmost node in the tree,
or the current node if it's the topmost node in the tree]
Returns:
[Node]: [the topmost Node in the tree]
"""
if isinstance(self, Document):
return self
node = self
nxt = self.parentNode
while nxt is not None:
node = nxt
nxt = nxt.parentNode
return node
[docs]
def insertBefore(self, new_node: Node, reference_node: Node | None = None) -> Node:
"""inserts a node before a reference node as a child of a specified parent node.
this will remove the node from its previous parent node, if any.
# TODO - can throw value error if wrong ordered params. may be helpful to catch to say so.
"""
if reference_node is None:
self.appendChild(new_node)
else:
# remove new_node from its previous parent node
if new_node.parentNode is not None:
new_node.parentNode.removeChild(new_node)
old_document = new_node.ownerDocument if isinstance(new_node.ownerDocument, Document) else None
previous_sibling = reference_node.previousSibling
self.args = (
self.args[: self.args.index(reference_node)]
+ (new_node,)
+ self.args[self.args.index(reference_node) :]
)
new_node.parentNode = self
_adopt_tree(new_node, old_document, self.ownerDocument if isinstance(self.ownerDocument, Document) else None)
_connect_tree(new_node)
_queue_mutation_record(
"childList",
self,
added_nodes=(new_node,),
previous_sibling=previous_sibling,
next_sibling=reference_node,
)
_notify_slot_change(self)
return new_node
[docs]
def removeChild(self, node: Node) -> Node | None:
"""removes a child node from the DOM and returns the removed node."""
for count, each in enumerate(self.args):
if type(each) == str:
continue
if each == node:
n = node
previous_sibling = n.previousSibling
next_sibling = n.nextSibling
_disconnect_tree(n)
n.parentNode = None
replace_args = list(self.args)
replace_args.remove(node)
self.args = tuple(replace_args)
_queue_mutation_record(
"childList",
self,
removed_nodes=(n,),
previous_sibling=previous_sibling,
next_sibling=next_sibling,
)
_notify_slot_change(self)
return n
r = each.removeChild(node)
if r:
return r
return None
[docs]
def replaceChild(self, newChild: "Node", oldChild: "Node") -> "Node":
"""[Replaces a child node within the given (parent) node.]
Args:
newChild ([type]): [a Node object]
oldChild ([type]): [a Node object]
Returns:
[type]: [the old child node]
"""
for count, each in enumerate(self.args):
if each == oldChild:
replace_args = list(self.args)
old_document = newChild.ownerDocument if isinstance(newChild.ownerDocument, Document) else None
previous_sibling = replace_args[count - 1] if count > 0 and isinstance(replace_args[count - 1], Node) else None
next_sibling = (
replace_args[count + 1] if count + 1 < len(replace_args) and isinstance(replace_args[count + 1], Node) else None
)
if isinstance(oldChild, Node):
_disconnect_tree(oldChild)
replace_args[count] = newChild
self.args = tuple(replace_args)
if isinstance(newChild, Node):
newChild.parentNode = self
_adopt_tree(newChild, old_document, self.ownerDocument if isinstance(self.ownerDocument, Document) else None)
_connect_tree(newChild)
if isinstance(oldChild, Node):
oldChild.parentNode = None
_queue_mutation_record(
"childList",
self,
added_nodes=(newChild,) if isinstance(newChild, Node) else (),
removed_nodes=(oldChild,) if isinstance(oldChild, Node) else (),
previous_sibling=previous_sibling,
next_sibling=next_sibling,
)
_notify_slot_change(self)
return oldChild
return oldChild
# for count, each in enumerate(self.args):
# if each == oldChild:
# n = oldChild
# self.removeChild(newChild) # doc remove child?
# list(self.args).remove(oldChild)
# list(self.args).insert(count, newChild)
# return n
# r = each.replaceChild(newChild, oldChild)
# if r:
# return r
# return None
[docs]
def cloneNode(self, deep: bool = True):
"""Returns a copy."""
import copy
if deep:
return copy.deepcopy(self)
else:
return copy.copy(self) # shallow copy
[docs]
def isSameNode(self, node):
"""Checks if two elements are the same node"""
return self == node
[docs]
def isEqualNode(self, node):
"""Checks if two elements are equal"""
return str(self) == str(node)
def getRootNode(self, options=None):
# if options is not None:
# if options['composed'] = True:
# TODO - need to implement composed
return self.rootNode
[docs]
def isDefaultNamespace(self, ns):
"""Checks if a namespace is the default namespace"""
if ns == self.namespaceURI:
return True
else:
return False
[docs]
def lookupNamespaceURI(self, ns: str):
"""Returns the namespace URI for a given prefix
:param ns: prefix - i.e 'xml', 'xlink', 'svg', etc
"""
from domonic.constants import namespaces
if ns in namespaces:
return namespaces[ns]
else:
return None
[docs]
def lookupPrefix(self, ns):
"""Returns the prefix for a given namespace URI"""
if ns == self.namespaceURI:
return self.prefix
else:
return None
@property
def nextSibling(self):
"""[returns the next sibling of the current node.]"""
if self.parentNode is None:
return None
else:
for count, node in enumerate(self.parentNode.args):
if node == self:
if count == len(self.parentNode.args) - 1:
return None
else:
return self.parentNode.args[count + 1]
[docs]
def normalize(self):
"""Normalize a node's value"""
return None
@property
def previousSibling(self):
"""[returns the previous sibling of the current node.]"""
if self.parentNode is None:
return None
else:
for count, node in enumerate(self.parentNode.args):
if node == self:
if count == 0:
return None
else:
return self.parentNode.args[count - 1]
@property
def textContent(self):
"""Returns the text content of a node and its descendants"""
# TODO - test- also check difference to nodeValue
# nodevalue is lvl 1 spec. textcontent is lvl 3 spec.
outp = ""
for each in self.args:
if type(each) is str:
outp = outp + each
else:
val = each.textContent
if val is not None:
outp = outp + val
else:
return None
if outp == "":
outp = None
return outp
@textContent.setter
def textContent(self, content):
"""Sets the text content of a node and its descendants"""
old_value = self.textContent
if content in (None, ""):
self.args = ()
else:
self.args = (content,)
if isinstance(self, CharacterData):
_queue_mutation_record("characterData", self, old_value=old_value)
return content
# def isSupported(self): return False # 🗑
# getUserData() 🗑️
# setUserData() 🗑️
# Non-standard helpers kept for etree-style compatibility.
# seems to make it work with https://github.com/sissaschool/elementpath
# if i hack it to allow domonic root nodes
[docs]
def iter(self, tag=None):
"""Creates a tree iterator with the current element as the root.
The iterator iterates over this element and all elements below it, in document (depth first) order.
If tag is not None or '*', only elements whose tag equals tag are returned from the iterator.
If the tree structure is modified during iteration, the result is undefined."""
for each in self.args:
if type(each) is str:
continue
if tag is None or tag == "*":
yield each
elif each.tag == tag:
yield each
for x in each.iter(tag):
yield x
@property
def tag(self):
"""Returns the tag name of the current node"""
return self.nodeName
# return self.tagName # not sure current is correct as would return #nodeName
@property
def text(self):
"""Returns the text content of the current node"""
return self.textContent
@property
def attrib(self):
"""Returns the attributes of the current node as a dict not a NamedNodeMap"""
try:
# print(self.kwargs)
return self.kwargs
except Exception as e:
# print('failed::', e)
return None
@property
def tail(self):
"""Returns the text content of the current node"""
return self.textContent
@property
def length(self) -> int:
return len(self)
[docs]
def is_matching(self, name, default_namespace=None):
"""
Determine if this node matches the given name and namespace.
"""
if name and name != self.tagName:
return False
if default_namespace and getattr(self, 'namespace', None) != default_namespace:
return False
return True
[docs]
class ParentNode:
"""not tested yet"""
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
# @property
# def childElementCount(self):
# return len(self.args)
@property
def children(self) -> "NodeList":
"""Return list of child nodes."""
return NodeList([e for e in self.childNodes if e.nodeType == Node.ELEMENT_NODE])
@property
def firstElementChild(self):
"""First Element child node."""
for child in self.childNodes:
if child.nodeType == Node.ELEMENT_NODE:
return child
return None
@property
def lastElementChild(self):
"""Last Element child node."""
for child in reversed(self.childNodes): # type: ignore
if child.nodeType == Node.ELEMENT_NODE:
return child
return None
def append(self, *args):
self.args += args
return self
def prepend(self, *args):
self.args = (args).extend(self.args)
return self
def replaceChildren(self, children):
self.args = children
[docs]
class ChildNode(Node):
"""not tested yet"""
[docs]
def remove(self):
"""Removes this ChildNode from the children list of its parent."""
if self.parentNode is None:
self._update_parents()
if self.parentNode is not None:
self.parentNode.removeChild(self)
return self
[docs]
def replaceWith(self, newChild):
"""Replaces this ChildNode with a new one."""
self.parentNode.replaceChild(newChild, self)
return self
[docs]
def before(self, newChild):
"""Inserts a newChild node immediately before this ChildNode."""
self.parentNode.insertBefore(newChild, self)
return self
[docs]
def after(self, newChild):
"""Inserts a newChild node immediately after this ChildNode."""
self.parentNode.insertBefore(newChild, self)
return self
[docs]
class Attr(Node):
# https://developer.mozilla.org/en-US/docs/Web/API/Attr
nodeType: int = Node.ATTRIBUTE_NODE
__slots__ = ("name", "value")
def __init__(self, name: str, value="", *args, **kwargs) -> None:
self.name: str = name
self.value = value
# self.nodeType: int = Node.ATTRIBUTE_NODE
@property
def isId(self) -> bool:
if self.name == "id":
return True
else:
return False
[docs]
def getNamedItem(self, name: str):
"""Returns a specified attribute node from a NamedNodeMap"""
for item in self.parentNode.attributes:
if item.name == name:
return item
return None
# def __getitem__(self, name):
# return self.getNamedItem(name)
# def __setitem__(self, name, value):
# self.setNamedItem(name, value)
[docs]
def removeNamedItem(self, name: str) -> bool:
"""Removes a specified attribute node"""
for item in self.parentNode.attributes:
if item.name == name:
self.parentNode.removeAttribute(item)
return True
return False
[docs]
def setNamedItem(self, name: str, value) -> bool:
"""Sets the specified attribute node (by name)"""
for item in self.parentNode.attributes:
if item.name == name:
item.value = value
return True
return False
[docs]
class NamedNodeMap:
"""Live attribute collection exposed by ``Element.attributes``.
``NamedNodeMap`` behaves like the DOM interface rather than a plain Python
dict: it is ordered, can be accessed by index or attribute name, and stays
in sync with the owning element's current attributes.
"""
def __init__(self, args: Iterable[Attr] | None = None, ownerDocument=None, parentNode=None):
self.parentNode = parentNode
self.ownerDocument = ownerDocument
self._attrs = list(args or [])
def _normalize_name(self, name: str) -> str:
return name[1:] if isinstance(name, str) and name.startswith("_") else name
def _storage_key(self, name: str) -> str:
normalized = self._normalize_name(name)
return normalized if normalized.startswith("_") else f"_{normalized}"
def _current_attrs(self) -> list[Attr]:
if self.parentNode is not None and hasattr(self.parentNode, "kwargs"):
return [Attr(key.lstrip("_"), value) for key, value in self.parentNode.kwargs.items()]
return list(self._attrs)
def _attribute_namespace(self, attr: Attr) -> str | None:
if ":" not in attr.name or self.parentNode is None:
return None
prefix = attr.name.split(":", 1)[0]
return self.parentNode.lookupNamespaceURI(prefix)
@property
def length(self) -> int:
return len(self._current_attrs())
def __len__(self) -> int:
return self.length
def __iter__(self) -> Iterator[Attr]:
return iter(self._current_attrs())
def __contains__(self, item: Any) -> bool:
if isinstance(item, Attr):
return self.getNamedItem(item.name) is not None
if isinstance(item, str):
return self.getNamedItem(item) is not None
return False
def __getitem__(self, key: int | str) -> Attr:
if isinstance(key, int):
item = self.item(key)
if item is None:
raise IndexError(key)
return item
item = self.getNamedItem(key)
if item is None:
raise KeyError(key)
return item
def __setitem__(self, key: str, value: Attr | Any) -> None:
if isinstance(value, Attr):
value.name = self._normalize_name(key)
self.setNamedItem(value)
return
self.setNamedItem(Attr(self._normalize_name(key), value))
def __delitem__(self, key: str) -> None:
removed = self.removeNamedItem(key)
if removed is None:
raise KeyError(key)
def item(self, index: int) -> Attr | None:
if not isinstance(index, int):
raise TypeError("index must be an integer")
attrs = self._current_attrs()
return attrs[index] if 0 <= index < len(attrs) else None
def getNamedItem(self, name: str) -> Attr | None:
normalized = self._normalize_name(name)
for item in self._current_attrs():
if item.name == normalized:
return item
return None
def setNamedItem(self, attr: Attr) -> Attr | None:
normalized = self._normalize_name(attr.name)
old_attr = self.getNamedItem(normalized)
attr.name = normalized
if self.parentNode is not None and hasattr(self.parentNode, "kwargs"):
self.parentNode.setAttribute(normalized, attr.value)
else:
self._attrs = [existing for existing in self._attrs if existing.name != normalized]
self._attrs.append(Attr(normalized, attr.value))
return old_attr
def removeNamedItem(self, name: str) -> Attr | None:
normalized = self._normalize_name(name)
old_attr = self.getNamedItem(normalized)
if old_attr is None:
return None
if self.parentNode is not None and hasattr(self.parentNode, "kwargs"):
self.parentNode.removeAttribute(normalized)
else:
self._attrs = [existing for existing in self._attrs if existing.name != normalized]
return old_attr
def getNamedItemNS(self, namespaceURI: str, localName: str) -> Attr | None:
normalized = self._normalize_name(localName)
for item in self._current_attrs():
item_local_name = item.name.split(":", 1)[-1]
if item_local_name == normalized and self._attribute_namespace(item) == namespaceURI:
return item
return None
def setNamedItemNS(self, attr: Attr) -> Attr | None:
return self.setNamedItem(attr)
def removeNamedItemNS(self, namespaceURI: str, localName: str) -> Attr | None:
attr = self.getNamedItemNS(namespaceURI, localName)
if attr is None:
return None
return self.removeNamedItem(attr.name)
def keys(self) -> list[str]:
return [attr.name for attr in self._current_attrs()]
def values(self) -> list[Attr]:
return self._current_attrs()
def items(self) -> list[tuple[str, Attr]]:
return [(attr.name, attr) for attr in self._current_attrs()]
[docs]
class DOMStringMap:
"""Dictionary-like helper for element dataset values."""
def __init__(self, *args, **kwargs):
self._store: dict[str, Any] = dict(*args, **kwargs)
super().__init__()
def __getitem__(self, name: str) -> Any:
return self._store[name]
def __setitem__(self, name: str, value: Any) -> None:
self._store[name] = value
def __contains__(self, name: str) -> bool:
return name in self._store
def __iter__(self):
return iter(self._store)
def __len__(self) -> int:
return len(self._store)
def keys(self):
return self._store.keys()
def values(self):
return self._store.values()
def items(self):
return self._store.items()
def __repr__(self) -> str:
return repr(self._store)
[docs]
def get(self, name: str):
"""Returns the value of the item with the specified name"""
return self._store.get(name)
[docs]
def set(self, name: str, value):
"""Sets the value of the item with the specified name"""
self._store[name] = value
return True
[docs]
def delete(self, name: str) -> bool:
"""Deletes the item with the specified name"""
if name in self._store:
del self._store[name]
return True
return False
# def has(self, name):
# """ Returns true if the specified name exists """
# for item in self.args:
# if item.name == name:
# return True
# return False
# def clear(self):
# """ Removes all items from the map """
# self.args = []
# return True
# def keys(self):
# """ Returns an array of all the names in the map """
# return [item.name for item in self.args]
# def values(self):
# """ Returns an array of all the values in the map """
# return [item.value for item in self.args]
[docs]
class DOMRectReadOnly:
"""Read-only rectangle object for DOM geometry APIs."""
@staticmethod
def fromRect(other: Any | None = None) -> "DOMRectReadOnly":
if other is None:
return DOMRectReadOnly()
return DOMRectReadOnly(
getattr(other, "x", 0),
getattr(other, "y", 0),
getattr(other, "width", 0),
getattr(other, "height", 0),
)
def __init__(self, x: float = 0, y: float = 0, width: float = 0, height: float = 0):
self._x = x
self._y = y
self._width = width
self._height = height
@property
def x(self) -> float:
return self._x
@property
def y(self) -> float:
return self._y
@property
def width(self) -> float:
return self._width
@property
def height(self) -> float:
return self._height
@property
def top(self) -> float:
return min(self._y, self._y + self._height)
@property
def right(self) -> float:
return max(self._x, self._x + self._width)
@property
def bottom(self) -> float:
return max(self._y, self._y + self._height)
@property
def left(self) -> float:
return min(self._x, self._x + self._width)
def toJSON(self):
return {
"x": self.x,
"y": self.y,
"width": self.width,
"height": self.height,
"top": self.top,
"right": self.right,
"bottom": self.bottom,
"left": self.left,
}
[docs]
class DOMRect(DOMRectReadOnly):
"""Mutable rectangle object for DOM geometry APIs."""
@staticmethod
def fromRect(other: Any | None = None) -> "DOMRect":
rect = DOMRectReadOnly.fromRect(other)
return DOMRect(rect.x, rect.y, rect.width, rect.height)
@DOMRectReadOnly.x.setter
def x(self, value: float) -> None:
self._x = value
@DOMRectReadOnly.y.setter
def y(self, value: float) -> None:
self._y = value
@DOMRectReadOnly.width.setter
def width(self, value: float) -> None:
self._width = value
@DOMRectReadOnly.height.setter
def height(self, value: float) -> None:
self._height = value
[docs]
class DOMRectList(list):
"""An ordered collection of DOMRect objects."""
@property
def length(self) -> int:
return len(self)
def item(self, index: int) -> DOMRect | None:
if not isinstance(index, int):
raise TypeError("index must be an integer")
return self[index] if 0 <= index < len(self) else None
[docs]
class DocumentTimeline:
"""Document-associated timeline used by animation surfaces.
This is the timing source behind ``document.timeline`` and
``Element.animate(...)``.
"""
def __init__(self, document: "Document | None" = None, originTime: float = 0.0):
self.document = document
self.originTime = float(originTime)
self._started_at = time.perf_counter()
@property
def currentTime(self) -> float:
return self.originTime + ((time.perf_counter() - self._started_at) * 1000.0)
[docs]
class CaretPosition:
"""Represents a caret location as a node plus offset pair."""
def __init__(self, offsetNode: Node | None = None, offset: int = 0) -> None:
self.offsetNode = offsetNode
self.offset = offset
def getClientRect(self) -> DOMRect:
if hasattr(self.offsetNode, "getBoundingClientRect"):
return self.offsetNode.getBoundingClientRect()
return DOMRect(0, 0, 0, 0)
[docs]
class Selection:
"""Represents the user's current selection within a document or shadow tree.
Domonic keeps both ordered ``Range`` data and anchor/focus information so
selection direction can still be represented.
"""
def __init__(self) -> None:
self._ranges: list[Range] = []
self._anchorNode: Node | None = None
self._anchorOffset: int = 0
self._focusNode: Node | None = None
self._focusOffset: int = 0
def _set_anchor_focus(
self,
anchorNode: Node | None,
anchorOffset: int = 0,
focusNode: Node | None = None,
focusOffset: int = 0,
) -> None:
self._anchorNode = anchorNode
self._anchorOffset = anchorOffset
self._focusNode = anchorNode if focusNode is None else focusNode
self._focusOffset = anchorOffset if focusNode is None else focusOffset
def _sync_anchor_focus_from_range(self, range_obj: "Range | None") -> None:
if range_obj is None:
self._set_anchor_focus(None, 0, None, 0)
return
self._set_anchor_focus(
range_obj.startContainer,
range_obj.startOffset,
range_obj.endContainer,
range_obj.endOffset,
)
@property
def rangeCount(self) -> int:
return len(self._ranges)
@property
def isCollapsed(self) -> bool:
return self.rangeCount == 0 or all(range_obj.collapsed for range_obj in self._ranges)
@property
def anchorNode(self) -> Node | None:
return self._anchorNode if self._ranges else None
@property
def anchorOffset(self) -> int:
return self._anchorOffset if self._ranges else 0
@property
def focusNode(self) -> Node | None:
return self._focusNode if self._ranges else None
@property
def focusOffset(self) -> int:
return self._focusOffset if self._ranges else 0
@property
def type(self) -> str:
if self.rangeCount == 0:
return "None"
return "Caret" if self.isCollapsed else "Range"
def addRange(self, range_obj: "Range") -> None:
if range_obj not in self._ranges:
self._ranges.append(range_obj)
if len(self._ranges) == 1:
self._sync_anchor_focus_from_range(range_obj)
def removeRange(self, range_obj: "Range") -> None:
self._ranges = [candidate for candidate in self._ranges if candidate is not range_obj]
self._sync_anchor_focus_from_range(self._ranges[0] if self._ranges else None)
def removeAllRanges(self) -> None:
self._ranges = []
self._sync_anchor_focus_from_range(None)
def getRangeAt(self, index: int) -> "Range":
if index < 0 or index >= len(self._ranges):
raise IndexError("Selection range index out of range")
return self._ranges[index]
def collapse(self, node: Node | None, offset: int = 0) -> None:
if node is None:
self.removeAllRanges()
return
range_obj = Range()
range_obj.setStart(node, offset)
range_obj.setEnd(node, offset)
self._ranges = [range_obj]
self._set_anchor_focus(node, offset, node, offset)
def collapseToStart(self) -> None:
if not self._ranges:
return
first = self._ranges[0]
self.collapse(first.startContainer, first.startOffset)
def collapseToEnd(self) -> None:
if not self._ranges:
return
last = self._ranges[-1]
self.collapse(last.endContainer, last.endOffset)
def extend(self, node: Node, offset: int = 0) -> None:
if not self._ranges:
self.collapse(node, offset)
return
anchor_node = self.anchorNode
anchor_offset = self.anchorOffset
active_range = self._ranges[-1]
if anchor_node is None:
active_range.setEnd(node, offset)
self._set_anchor_focus(active_range.startContainer, active_range.startOffset, node, offset)
return
if Range._compare_points(anchor_node, anchor_offset, node, offset) <= 0:
active_range.setStart(anchor_node, anchor_offset)
active_range.setEnd(node, offset)
else:
active_range.setStart(node, offset)
active_range.setEnd(anchor_node, anchor_offset)
self._set_anchor_focus(anchor_node, anchor_offset, node, offset)
def setBaseAndExtent(
self,
anchorNode: Node,
anchorOffset: int,
focusNode: Node,
focusOffset: int,
) -> None:
range_obj = Range()
if Range._compare_points(anchorNode, anchorOffset, focusNode, focusOffset) <= 0:
range_obj.setStart(anchorNode, anchorOffset)
range_obj.setEnd(focusNode, focusOffset)
else:
range_obj.setStart(focusNode, focusOffset)
range_obj.setEnd(anchorNode, anchorOffset)
self._ranges = [range_obj]
self._set_anchor_focus(anchorNode, anchorOffset, focusNode, focusOffset)
def empty(self) -> None:
self.removeAllRanges()
def selectAllChildren(self, node: "Node") -> None:
range_obj = Range()
range_obj.selectNodeContents(node)
self._ranges = [range_obj]
self._set_anchor_focus(node, 0, node, Range._container_length(node))
def deleteFromDocument(self) -> None:
for range_obj in list(self._ranges):
range_obj.deleteContents()
self.removeAllRanges()
def containsNode(self, node: Node | None, allowPartialContainment: bool = False) -> bool:
if node is None:
return False
if isinstance(node, Text):
node_end_offset = len(node.textContent)
else:
node_end_offset = len(list(getattr(node, "childNodes", [])))
for range_obj in self._ranges:
start_relation = range_obj.comparePoint(node, 0)
end_relation = range_obj.comparePoint(node, node_end_offset)
if allowPartialContainment:
if start_relation != 1 and end_relation != -1:
return True
else:
if start_relation == 0 and end_relation == 0:
return True
return False
def toString(self) -> str:
return "".join(range_obj.toString() for range_obj in self._ranges)
__str__ = toString
[docs]
class DOMTokenList(list):
"""DOMTokenList represents a set of space-separated tokens."""
def __init__(self, element: "Node"):
self.el = element
# trim and split on whitespace
# classes = element.className.replace(r'^\s+|\s+$/g', '').split(r'\s+/')
self.classes = element.className.split(" ")
self.classes = [x.strip() for x in self.classes]
super().__init__(self.classes)
[docs]
def add(self, *args):
"""Adds the given tokens to the list"""
for item in args:
if item not in self:
self.append(item)
self.el.className = self.toString()
[docs]
def remove(self, *args):
"""Removes the given tokens from the list"""
for item in args:
if item in self:
super().remove(item)
self.el.className = self.toString()
[docs]
def toggle(self, token, force=None):
"""If force is not given, removes token from list if present,
otherwise adds token to list. If force is true, adds token to list,
and if force is false, removes token from list if present."""
if force is None:
if token in self:
self.remove(token)
else:
self.add(token)
elif force is True:
self.add(token)
elif force is False:
self.remove(token)
else:
raise TypeError("force must be a boolean")
self.classes = list(self)
[docs]
def contains(self, token) -> bool:
"""Returns true if the token is in the list, and false otherwise"""
# return token in self.el.className
return token in self
[docs]
def item(self, index: int):
"""Returns the token at the specified index"""
return self[index] if 0 <= index < len(self) else None
[docs]
def toString(self) -> str:
"""Returns a string containing all tokens in the list, with spaces separating each token"""
return " ".join(self)
def __str__(self):
return self.toString()
[docs]
class ShadowRoot(Node): # TODO - this may need to extend tag also to get the args/kwargs
"""property on element that has hidden DOM"""
def __init__(self, host, mode="open"):
self.adoptedStyleSheets = []
self.delegatesFocus = False
self.host = host
self.mode = mode
self.parentNode = host
self._selection = Selection()
super().__init__()
[docs]
def elementFromPoint(self, x: float, y: float) -> Element | None:
"""Returns the topmost element at the specified coordinates."""
hits = self.elementsFromPoint(x, y)
return hits[0] if hits else None
[docs]
def getSelection(self) -> Selection:
"""Returns a Selection object for the document."""
if not hasattr(self, "_selection"):
self._selection = Selection()
return self._selection
[docs]
def elementsFromPoint(self, x: float, y: float) -> list[Element]:
"""Returns an array of all elements at the specified coordinates."""
matches = []
def walk(node):
if not isinstance(node, Element):
return
rect = node.getBoundingClientRect()
if rect.left <= x <= rect.right and rect.top <= y <= rect.bottom:
matches.append(node)
for child in getattr(node, "childNodes", []):
walk(child)
for child in self.childNodes:
walk(child)
return matches
[docs]
def caretPositionFromPoint(self, x: float = 0, y: float = 0) -> CaretPosition | None:
"""
Returns a CaretPosition object containing the DOM node containing the caret,
and caret's character offset within that node.
"""
target = self.elementFromPoint(x, y)
if target is None:
return None
first_child = target.firstChild
if isinstance(first_child, Text):
rect = target.getBoundingClientRect()
width = max(rect.width, 1)
text_length = len(first_child.textContent)
relative = max(0, min(x - rect.left, width))
offset = min(text_length, int((relative / width) * text_length))
return CaretPosition(first_child, offset)
return CaretPosition(target, 0)
[docs]
class DocumentType(Node):
nodeType = Node.DOCUMENT_TYPE_NODE
__slots__ = ("name", "publicId", "systemId")
def __init__(self, name: str = "html", publicId: str = "", systemId: str = "") -> None:
self.name: str = name # A DOMString, eg "html" for <!DOCTYPE HTML>.
self.publicId: str = publicId # eg "-//W3C//DTD HTML 4.01//EN", empty string for HTML5.
self.systemId: str = systemId # eg "http://www.w3.org/TR/html4/strict.dtd", empty string for HTML5.
super().__init__()
[docs]
def internalSubset(self):
"""A DOMString of the internal subset, or None. Eg "<!ELEMENT foo (bar)>"."""
if self.systemId:
return self.systemId
else:
return None
[docs]
def notations(self) -> NamedNodeMap:
"""A NamedNodeMap with notations declared in the DTD."""
nnm = NamedNodeMap()
for item in self.ownerDocument.args:
if item.nodeType == Node.NOTATION_NODE:
nnm.append(item)
return nnm
def __str__(self) -> str:
# return f"<!DOCTYPE {self.name} {self.publicId} {self.systemId}>"
# TODO fix broken spacing when no publicId or systemId
full_str = f"<!DOCTYPE {self.name}"
if self.publicId:
full_str += f" PUBLIC {self.publicId}"
if self.systemId:
full_str += f" SYSTEM {self.systemId}"
full_str += ">"
return full_str
"""
def AriaMixin(): # ???
@property
def ariaAtomic(self):
return self.getAttribute('aria-atomic')
@ariaAtomic.setter
def ariaAtomic(self, value: str):
self.setAttribute('aria-atomic', value)
@property
def ariaAtomic(self):
return self.getAttribute('aria-atomic')
@property
def ariaAutoComplete(self):
return self.getAttribute('aria-autoComplete')
@ariaAutoComplete.setter
def ariaAutoComplete(self, value: str):
return self.getAttribute('aria-autoComplete')
@property
def ariaBusy(self):
return self.getAttribute('aria-busy')
@ariaBusy.setter
def ariaBusy(self, value: str):
return self.getAttribute('aria-busy')
@property
def ariaChecked(self):
return self.getAttribute('aria-checked')
@ariaChecked.setter
def ariaChecked(self, value: str):
return self.getAttribute('aria-checked')
@property
def ariaColCount(self):
return self.getAttribute('aria-colCount')
@ariaColCount.setter
def ariaColCount(self, value: str):
return self.getAttribute('aria-colCount')
@property
def ariaColIndex(self):
return self.getAttribute('aria-colIndex')
@ariaColIndex.setter
def ariaColIndex(self, value: str):
return self.getAttribute('aria-colIndex')
@property
def ariaColIndexText(self):
return self.getAttribute('aria-colIndexText')
@ariaColIndexText.setter
def ariaColIndexText(self, value: str):
return self.getAttribute('aria-colIndexText')
@property
def ariaColSpan(self):
return self.getAttribute('aria-colSpan')
@ariaColSpan.setter
def ariaColSpan(self, value: str):
return self.getAttribute('aria-colSpan')
@property
def ariaCurrent(self):
return self.getAttribute('aria-current')
@ariaCurrent.setter
def ariaCurrent(self, value: str):
return self.getAttribute('aria-current')
@property
def ariaDescription(self):
return self.getAttribute('aria-description')
@ariaDescription.setter
def ariaDescription(self, value: str):
return self.getAttribute('aria-description')
@property
def ariaDisabled(self):
return self.getAttribute('aria-disabled')
@ariaDisabled.setter
def ariaDisabled(self, value: str):
return self.getAttribute('aria-disabled')
@property
def ariaExpanded(self):
return self.getAttribute('aria-expanded')
@ariaExpanded.setter
def ariaExpanded(self, value: str):
return self.getAttribute('aria-expanded')
@property
def ariaHasPopup(self):
return self.getAttribute('aria-hasPopup')
@ariaHasPopup.setter
def ariaHasPopup(self, value: str):
return self.getAttribute('aria-hasPopup')
@property
def ariaHidden(self):
return self.getAttribute('aria-hidden')
@ariaHidden.setter
def ariaHidden(self, value: str):
return self.getAttribute('aria-hidden')
@property
def ariaKeyShortcuts(self):
return self.getAttribute('aria-keyShortcuts')
@ariaKeyShortcuts.setter
def ariaKeyShortcuts(self, value: str):
return self.getAttribute('aria-keyShortcuts')
@property
def ariaLabel(self):
return self.getAttribute('aria-label')
@ariaLabel.setter
def ariaLabel(self, value: str):
return self.getAttribute('aria-label')
@property
def ariaLevel(self):
return self.getAttribute('aria-level')
@ariaLevel.setter
def ariaLevel(self, value: str):
return self.getAttribute('aria-level')
@property
def ariaLive(self):
return self.getAttribute('aria-live')
@ariaLive.setter
def ariaLive(self, value: str):
return self.getAttribute('aria-live')
@property
def ariaModal(self):
return self.getAttribute('aria-modal')
@ariaModal.setter
def ariaModal(self, value: str):
return self.getAttribute('aria-modal')
@property
def ariaMultiline(self):
return self.getAttribute('aria-multiline')
@ariaMultiline.setter
def ariaMultiline(self, value: str):
return self.getAttribute('aria-multiline')
@property
def ariaMultiSelectable(self):
return self.getAttribute('aria-multiSelectable')
@ariaMultiSelectable.setter
def ariaMultiSelectable(self, value: str):
return self.getAttribute('aria-multiSelectable')
@property
def ariaOrientation(self):
return self.getAttribute('aria-orientation')
@ariaOrientation.setter
def ariaOrientation(self, value: str):
return self.getAttribute('aria-orientation')
@property
def ariaPlaceholder(self):
return self.getAttribute('aria-placeholder')
@ariaPlaceholder.setter
def ariaPlaceholder(self, value: str):
return self.getAttribute('aria-placeholder')
@property
def ariaPosInSet(self):
return self.getAttribute('aria-posInSet')
@ariaPosInSet.setter
def ariaPosInSet(self, value: str):
return self.getAttribute('aria-posInSet')
@property
def ariaPressed(self):
return self.getAttribute('aria-pressed')
@ariaPressed.setter
def ariaPressed(self, value: str):
return self.getAttribute('aria-pressed')
@property
def ariaReadOnly(self):
return self.getAttribute('aria-readOnly')
@ariaReadOnly.setter
def ariaReadOnly(self, value: str):
return self.getAttribute('aria-readOnly')
@property
def ariaRelevant(self):
return self.getAttribute('aria-relevant')
@ariaRelevant.setter
def ariaRelevant(self, value: str):
return self.getAttribute('aria-relevant')
@property
def ariaRequired(self):
return self.getAttribute('aria-required')
@ariaRequired.setter
def ariaRequired(self, value: str):
return self.getAttribute('aria-required')
@property
def ariaRoleDescription(self):
return self.getAttribute('aria-roleDescription')
@ariaRoleDescription.setter
def ariaRoleDescription(self, value: str):
return self.getAttribute('aria-roleDescription')
@property
def ariaRowCount(self):
return self.getAttribute('aria-rowCount')
@ariaRowCount.setter
def ariaRowCount(self, value: str):
return self.getAttribute('aria-rowCount')
@property
def ariaRowIndex(self):
return self.getAttribute('aria-rowIndex')
@ariaRowIndex.setter
def ariaRowIndex(self, value: str):
return self.getAttribute('aria-rowIndex')
@property
def ariaRowIndexText(self):
return self.getAttribute('aria-rowIndexText')
@ariaRowIndexText.setter
def ariaRowIndexText(self, value: str):
return self.getAttribute('aria-rowIndexText')
@property
def ariaRowSpan(self):
return self.getAttribute('aria-rowSpan')
@ariaRowSpan.setter
def ariaRowSpan(self, value: str):
return self.getAttribute('aria-rowSpan')
@property
def ariaSelected(self):
return self.getAttribute('aria-selected')
@ariaSelected.setter
def ariaSelected(self, value: str):
return self.getAttribute('aria-selected')
@property
def ariaSetSize(self):
return self.getAttribute('aria-setSize')
@ariaSetSize.setter
def ariaSetSize(self, value: str):
return self.getAttribute('aria-setSize')
@property
def ariaSort(self):
return self.getAttribute('aria-sort')
@ariaSort.setter
def ariaSort(self, value: str):
return self.getAttribute('aria-sort')
@property
def ariaValueMax(self):
return self.getAttribute('aria-valueMax')
@ariaValueMax.setter
def ariaValueMax(self, value: str):
return self.getAttribute('aria-valueMax')
@property
def ariaValueMin(self):
return self.getAttribute('aria-valueMin')
@ariaValueMin.setter
def ariaValueMin(self, value: str):
return self.getAttribute('aria-valueMin')
@property
def ariaValueNow(self):
return self.getAttribute('aria-valueNow')
@ariaValueNow.setter
def ariaValueNow(self, value: str):
return self.getAttribute('aria-valueNow')
@property
def ariaValueText(self):
return self.getAttribute('aria-valueText')
@ariaValueText.setter
def ariaValueText(self, value: str):
return self.getAttribute('aria-valueText')
# class ElementInternals(object, AriaMixin):
# def __init__(self, element):
# self.element = element
# self.shadowRoot = None # Returns the ShadowRoot object associated with this element.
# self.form # Returns the HTMLFormElement associated with this element.
# self.states # Returns the CustomStateSet associated with this element.
# self.willValidate # A boolean value which returns true if the element is a submittable element that is a candidate for constraint validation.
# self.validity # Returns a ValidityState object which represents the different validity states the element can be in, with respect to constraint validation.
# self.validationMessage # A string containing the validation message of this element.
# self.labels # Returns a NodeList of all of the label elements associated with this element.
class CustomStateSet:
def __init__(self):
pass
def add(self, state):
pass
def clear(self):
pass
def delete(self, state):
pass
"""
[docs]
class NodeList(list):
"""NodeList objects are collections of nodes"""
@property
def length(self) -> int:
return len(self)
[docs]
def item(self, index: int) -> Node | None:
"""Returns an item in the list by its index, or null if the index is out-of-bounds."""
# An alternative to accessing nodeList[i] (which instead returns undefined when i is out-of-bounds).
# This is mostly useful for non-JavaScript DOM implementations.
try:
return self[index] if 0 <= index < self.length else None
except IndexError:
return None
# def items(self):
# """ Returns a list of the nodes in the list."""
# return self
[docs]
def entries(self) -> Iterable[tuple[int, Node]]:
"""Returns an iterator, allowing code to go through all key/value pairs contained in the collection.
(In this case, the keys are numbers starting from 0 and the values are nodes."""
# i.e. Array [ 0, <p> ]
for i in range(len(self)):
yield i, self[i]
[docs]
def forEach(self, func: Callable[[Node, int, "NodeList"], Any], thisArg: Any = None) -> None:
"""Calls a function for each item in the NodeList."""
# thisArg = thisArg or self
for i in range(len(self)):
func(self[i], i, self)
[docs]
def keys(self) -> Iterable[int]:
"""Returns an iterator, allowing code to go through all the keys of the key/value pairs contained in the collection.
(In this case, the keys are numbers starting from 0.)"""
return iter(range(len(self)))
[docs]
def values(self) -> Iterable[Node]:
"""Returns an iterator allowing code to go through all values (nodes) of the key/value pairs
contained in the collection."""
return iter(self)
[docs]
class RadioNodeList(NodeList):
# TODO - not tested
def __init__(self, name: str) -> None: # , owner: Element):
self.name: str = name
def __iter__(self) -> Iterator[Node]:
return iter(self.getElementsByName(self.name))
def __getitem__(self, index: int) -> Node:
return self.getElementsByName(self.name)[index]
def __len__(self) -> int:
return len(self.getElementsByName(self.name))
@property
def value(self) -> Any:
"""Returns the value of the first element in the collection,
or null if there are no elements in the collection."""
return self[0].value if len(self) > 0 else None
[docs]
class Element(Node):
"""Baseclass for all html tags"""
# __slots__ = ('_id')
def __init__(self, *args, **kwargs):
# self.content = None
# self.attributes = None
if self.hasAttribute("id"):
self.id = self.id # ''#None
self.lang = None
self.tabIndex = None
if self.hasAttribute("title"):
self.title = self.title
if self.hasAttribute("class"):
self.className = self.className
self.classList = self.classList
# self.tagName
self.style = None # Style(self) # = #'test'#Style()
self.shadowRoot = None
self.dir = None
super().__init__(*args, **kwargs)
def _getElementById(self, _id: str):
# TODO - i think i need to build a hash map of IDs to positions on the tree
# for now I'm going using recursion so this is a bit of a hack to do a few levels
if self.getAttribute("id") == _id:
return self
for child in self.childNodes:
if not isinstance(child, Element):
continue
match = child._getElementById(_id)
if match is not False and match is not None:
return match
return False
def _getElementByAttrVal(self, attr: str, val: str):
# TODO - i think i need to build a hash map of IDs to positions on the tree
# for now I'm going using recursion so this is a bit of a hack to do a few levels
if self.getAttribute(attr) == val:
return self
try:
for child in self.childNodes:
match = child._getElementByAttrVal(attr, val)
if match:
return match
except Exception as e:
pass # TODO - dont iterate strings
return False
def _matchElement(self, element, query):
"""
tries to match an element based on the query
at moment very basic. i.e. single level. just checks between id/tag/class
"""
if not isinstance(element, Element):
return False
query = query.strip()
if not query:
return False
if "." in query and not query.startswith(".") and "[" not in query:
tag_name, class_name = query.split(".", 1)
return element.tagName.lower() == tag_name.lower() and class_name in element.classList
if "#" in query and not query.startswith("#") and "[" not in query:
tag_name, element_id = query.split("#", 1)
return element.tagName.lower() == tag_name.lower() and element.getAttribute("id") == element_id
if query[0] == "#":
if element.getAttribute("id") == query.split("#")[1]:
return True
if query == "*":
return True
if element.tagName.lower() == query.lower():
return True
if query[0] == ".":
if query.split(".")[1] in element.classList:
return True
return False
[docs]
def matches(self, s: str) -> bool:
"""[checks to see if the Element would be selected by the provided selectorString]
https://developer.mozilla.org/en-US/docs/Web/API/Element/matches
Args:
s (str): [css selector]
Returns:
[bool]: [True if selector maches Element otherwise False]
"""
if self.ownerDocument is None:
return False
matches = self.ownerDocument.querySelectorAll(s)
for match in matches:
if match == self:
return True
return False
# https://developer.mozilla.org/en-US/docs/Web/API/Element/closest
def closest(self, s: str):
el = self
while el != None and el.nodeType == 1: # TODO - nodeType
if Element.matches(el, s):
return el
el = el.parentElement or el.parentNode
return None
# @staticmethod
[docs]
def getElementsBySelector(self, all_selectors, document):
"""
Get DOM elements based on the given CSS Selector.
Original sources:
- https://simonwillison.net/2003/Mar/25/getElementsBySelector/
- http://www.openjs.com/scripts/dom/css_selector/
- https://bin-co.com/python/scripts/getelementsbyselector-html-css-query.php (ported to Python 2, broken/bugs, BSD licensed)
Note:
- Always include a tag in the query. For example, `'a.classname'` will work, but just `'.classname'` will not.
- Preserved as a compatibility helper for older selector-style code.
- TODO: Needs to work in conjunction with `_matchElement` for better querySelector support, and to ensure dQuery compatibility.
- TODO: Implement support for `*=` (node content).
Args:
all_selectors (str): The CSS selectors to query.
document (object): The document object to search within.
Returns:
list: A list of elements matching the CSS selectors.
"""
if not all_selectors:
return []
selected = []
selectors = [selector.strip() for selector in str(all_selectors).split(",") if selector.strip()]
if len(selectors) > 1:
seen = []
for selector in selectors:
for item in self.getElementsBySelector(selector, document):
if item not in seen:
seen.append(item)
return seen
current = []
quote = None
for char in all_selectors:
if quote is not None:
current.append(char)
if char == quote:
quote = None
continue
if char in ("'", '"'):
quote = char
current.append(char)
continue
if char.isspace():
if current and current[-1] != " ":
current.append(" ")
continue
current.append(char)
all_selectors = "".join(current).strip()
# Grab all of the tagName elements within current context
def getElements(context, tag):
if tag == "":
tag = "*"
# Get elements matching tag, filter them for class selector
found = []
for con in context:
elements = con.getElementsByTagName(tag)
found.extend(elements)
return found
context = [document]
inheriters = []
current = []
bracket_depth = 0
quote = None
for char in all_selectors:
if quote is not None:
current.append(char)
if char == quote:
quote = None
continue
if char in ("'", '"'):
quote = char
current.append(char)
continue
if char == "[":
bracket_depth += 1
current.append(char)
continue
if char == "]":
bracket_depth = max(0, bracket_depth - 1)
current.append(char)
continue
if char == " " and bracket_depth == 0:
if current:
inheriters.append("".join(current))
current = []
continue
current.append(char)
if current:
inheriters.append("".join(current))
# Space
for element in inheriters:
# This part is to make sure that it is not part of a CSS3 Selector
left_bracket = str.find(element, "[")
right_bracket = str.find(element, "]")
pos = str.find(element, "#") # ID
if pos + 1 and not (pos > left_bracket and pos < right_bracket):
parts = str.split(element, "#")
tag = parts[0]
id = parts[1]
ele = document.getElementById(id)
context = [ele] # [](ele)
continue
pos = str.find(element, ".") # Class
if pos + 1 and not (pos > left_bracket and pos < right_bracket):
parts = str.split(element, ".")
tag = parts[0]
class_name = parts[1]
found = getElements(context, tag)
# found = document.getElementsByClassName(class_name)
context = []
for fnd in found:
if fnd.getAttribute("class") and re.search(
r"(^|\s)" + class_name + r"(\s|$)", fnd.getAttribute("class")
):
context.append(fnd)
continue
# If the char '[' appears, that means it needs CSS 3 parsing
if str.find(element, "[") + 1:
# Code to deal with attribute selectors
m = re.match(r"^([\w\*-]*)\[([\w-]+)([=~\|\^\$\*]?)=?['\"]?([^\]'\"]*)['\"]?\]$", element)
if m:
tag = m.group(1)
attr = m.group(2)
operator = m.group(3)
value = m.group(4)
else:
return []
found = getElements(context, tag)
context = []
for fnd in found:
attr_value = fnd.getAttribute(attr)
if attr_value is None:
continue
if operator == "=" and fnd.getAttribute(attr) != value:
continue # WORKING
if operator == "~" and not (re.search(r"(^|\\s)" + value + "(\\s|$)", attr_value)):
continue # NOT WORKING?
if operator == "|" and not (re.search(r"^" + value + "-?", attr_value)):
continue
if operator == "^" and str.find(attr_value, value) != 0:
continue # WORKING
if operator == "$" and str.rfind(attr_value, value) != (len(attr_value) - len(value)):
continue # kinda WORKING
if operator == "*" and not (str.find(attr_value, value) + 1):
continue # WORKING
elif not attr_value:
continue
context.append(fnd)
continue
# Tag selectors - no class or id specified.
found = getElements(context, element)
context = found
selected.extend(context)
return selected
[docs]
def append(self, *args):
"""Inserts a set of Node objects or DOMString objects after the last child of the Element."""
self.args += args
self._update_parents()
return self
# elem.attachShadow({mode: open|closed})
def attachShadow(self, obj):
mode = (obj or {}).get("mode", "open")
self.shadowRoot = ShadowRoot(self, mode)
return self.shadowRoot
# def accessKey( key: str ): -> None
# ''' Sets or returns the accesskey attribute of an element'''
# return
# example
# dom.getElementById("myAnchor").accessKey = "w";
@property
def attributes(self) -> NamedNodeMap:
"""Returns a NamedNodeMap of an element's attributes"""
newargs: list = []
for key, value in self.kwargs.items():
newargs.append(Attr(key.lstrip("_"), value))
nnm = NamedNodeMap(newargs, None, self)
return nnm
@property
def innerHTML(self):
"""Sets or returns the content of an element"""
return self.content
@innerHTML.setter
def innerHTML(self, value):
if value is not None:
self.args = tuple(self._parse_html_fragment(value))
self._update_parents()
return self.content
@property
def outerHTML(self):
return self
@outerHTML.setter
def outerHTML(self, value):
if isinstance(value, Element):
replacement_nodes = [value]
elif isinstance(value, str):
replacement_nodes = self._parse_html_fragment(value)
else:
replacement_nodes = [value]
if self.parentNode is None:
return self
parent = self.parentNode
index = parent.args.index(self)
parent.args = parent.args[:index] + tuple(replacement_nodes) + parent.args[index + 1 :]
parent._update_parents()
return self
def html(self, *args):
self.args = args
return self
def _parse_html_fragment(self, value: Any) -> list[Any]:
if isinstance(value, Element):
return [value]
if isinstance(value, (list, tuple)):
nodes: list[Any] = []
for item in value:
nodes.extend(self._parse_html_fragment(item))
return nodes
if not isinstance(value, str):
return [value]
if "<" not in value or ">" not in value:
return [value]
try:
from domonic import domonic as domonic_module
parsed = domonic_module.parseString(f"<div>{value}</div>")
wrapper = parsed.querySelector("div") if parsed is not None else None
if wrapper is None:
return [value]
return list(wrapper.args)
except Exception:
return [value]
[docs]
def blur(self):
"""Removes focus from an element"""
from domonic.events import FocusEvent
doc = self.ownerDocument if isinstance(self.ownerDocument, Document) else None
self._focused = False
related_target = None
if doc is not None and getattr(doc, "_activeElement", None) is self:
doc._activeElement = None
related_target = getattr(doc, "body", None)
result = self.dispatchEvent(FocusEvent("blur", {"bubbles": False, "cancelable": False, "relatedTarget": related_target}))
self.dispatchEvent(FocusEvent("focusout", {"bubbles": True, "cancelable": False, "relatedTarget": related_target}))
return result
@property
def classList(self):
"""Returns the value of the classList attribute of an element"""
cl = self.getAttribute("class")
if cl is None:
return [] # TODO - fix this
else:
return DOMTokenList(self)
@classList.setter
def classList(self, newlist):
"""Sets or returns the value of the classList attribute of an element"""
self.setAttribute("class", newlist)
# raise NotImplementedError
@property
def className(self):
"""Sets or returns the value of the className attribute of an element"""
return self.getAttribute("class")
@className.setter
def className(self, newname: str):
"""Sets or returns the value of the className attribute of an element"""
self.setAttribute("class", newname)
[docs]
def click(self):
"""Simulates a mouse-click on an element"""
view = getattr(self.ownerDocument, "defaultView", None) if isinstance(self.ownerDocument, Document) else None
evt = MouseEvent("click", {"bubbles": True, "cancelable": True, "view": view, "detail": 1})
return self.dispatchEvent(evt)
def animate(self, keyframes: list[dict[str, Any]] | dict[str, Any], options: Any = None):
from domonic.animation import Animation, KeyframeEffect
owner_document = self.ownerDocument if isinstance(self.ownerDocument, Document) else globals().get("document")
timeline = owner_document.timeline if isinstance(owner_document, Document) else None
effect = KeyframeEffect(self, keyframes, options)
animation = Animation(effect, timeline)
animation.play()
return animation
@staticmethod
def _style_number(value):
if value in (None, "", "auto", "none"):
return 0
if isinstance(value, (int, float)):
return value
match = re.search(r"-?\d+(?:\.\d+)?", str(value))
return float(match.group(0)) if match else 0
@property
def clientHeight(self):
"""Returns the height of an element, including padding"""
return (
Element._style_number(self.style.height)
+ Element._style_number(self.style.paddingTop)
+ Element._style_number(self.style.paddingBottom)
)
@property
def clientLeft(self):
"""Returns the width of the left border of an element"""
return Element._style_number(self.style.left)
@property
def clientTop(self):
"""Returns the width of the top border of an element"""
return Element._style_number(self.style.top)
@property
def clientWidth(self):
"""Returns the width of an element, including padding"""
return (
Element._style_number(self.style.width)
+ Element._style_number(self.style.paddingLeft)
+ Element._style_number(self.style.paddingRight)
)
@property
def contentEditable(self) -> bool:
"""Sets or returns whether an element is editable"""
is_editable = self.getAttribute("contenteditable")
return True if (is_editable == "true" or is_editable is True) else False
@contentEditable.setter
def contentEditable(self, value: bool) -> None:
self.setAttribute("contenteditable", value)
@property
def dataset(self):
"""Returns the value of the dataset attribute of an element"""
from domonic.utils import Utils
dsmap = DOMStringMap()
for key, value in self.kwargs.items():
attr_name = key[1:] if key.startswith("_") else key
if attr_name.startswith("data-"):
dsmap[Utils.case_camel(attr_name.replace("data-", ""))] = value
return dsmap
@property
def dir(self):
"""returns the value of the dir attribute of an element"""
return self.getAttribute("dir")
@dir.setter
def dir(self, direction: str = "auto"):
"""Sets the value of the dir attribute of an element"""
self.setAttribute("dir", direction)
[docs]
def exitFullscreen(self):
"""Cancels an element in fullscreen mode"""
doc = self.ownerDocument
if doc is not None:
doc._fullscreenElement = None
return None
[docs]
def firstElementChild(self):
"""Returns the first child element of an element"""
try:
return self.args[0]
except Exception:
return None
[docs]
def focus(self):
"""Sets focus on an element"""
from domonic.events import FocusEvent
doc = self.ownerDocument if isinstance(self.ownerDocument, Document) else None
previous = None
if doc is not None:
current = getattr(doc, "_activeElement", None)
if current is not None and current is not self:
previous = current
current._focused = False
current.dispatchEvent(FocusEvent("blur", {"bubbles": False, "cancelable": False, "relatedTarget": self}))
current.dispatchEvent(FocusEvent("focusout", {"bubbles": True, "cancelable": False, "relatedTarget": self}))
doc._activeElement = self
self._focused = True
result = self.dispatchEvent(FocusEvent("focus", {"bubbles": False, "cancelable": False, "relatedTarget": previous}))
self.dispatchEvent(FocusEvent("focusin", {"bubbles": True, "cancelable": False, "relatedTarget": previous}))
return result
[docs]
def setAttributeNodeNS(self, attr):
"""Sets the attribute node of an element"""
a = Attr(attr.name.lstrip("_"), attr.value)
self.setAttributeNode(a)
return self
[docs]
def getAttributeNodeNS(self, attr):
"""Sets the attribute node of an element"""
a = self.getAttribute(attr)
if a is None:
return None
return Attr(attr, a)
[docs]
def setAttributeNS(self, namespaceURI, localName, value):
"""Sets an attribute in the given namespace"""
self.setAttribute(localName, value)
[docs]
def getAttributeNS(self, namespaceURI, localName):
"""Returns the value of the specified attribute"""
return self.getAttribute(localName)
[docs]
def removeAttributeNS(self, namespaceURI, localName):
"""Removes an attribute from an element"""
if localName in self.attributes:
self.removeAttribute(localName)
# else:
# raise AttributeError
return self
[docs]
def getAttribute(self, attribute: str) -> str:
"""Returns the specified attribute value of an element node"""
try:
if attribute[0:1] != "_":
attribute = "_" + attribute
return self.kwargs[attribute]
except KeyError:
return None
[docs]
def getAttributeNode(self, attribute: str) -> str:
"""Returns the specified attribute node"""
value = self.getAttribute(attribute)
if value is None:
return None
return Attr(attribute.lstrip("_"), value)
[docs]
def getBoundingClientRect(self):
"""Returns the size of an element and its position relative to the viewport"""
rect = DOMRect(
Element._style_number(self.style.left),
Element._style_number(self.style.top),
self.offsetWidth(),
self.offsetHeight(),
)
_process_observer_notifications(self, rect)
return rect
[docs]
def getSelection(self):
"""Returns a Selection object for this element's root tree."""
root = self.rootNode
if hasattr(root, "_selection"):
return root._selection
selection = Selection()
setattr(root, "_selection", selection)
return selection
[docs]
def getElementsByClassName(self, className: str) -> "HTMLCollection":
"""[Returns a collection of all child elements with the specified class name]
Args:
className (str): [a DOMString representing the class name to match]
Returns:
[type]: [a NodeList of all child elements with the specified class name]
"""
# TODO - this will have to change as this i live and qsa aint.
# return self.querySelectorAll('.' + className)
return HTMLCollection(self.querySelectorAll("." + className))
[docs]
def elementFromPoint(self, x: float, y: float) -> Element | None:
"""Returns the topmost element in this subtree at the specified coordinates."""
hits = self.elementsFromPoint(x, y)
return hits[0] if hits else None
[docs]
def elementsFromPoint(self, x: float, y: float) -> list[Element]:
"""Returns all elements in this subtree at the specified coordinates."""
matches = []
def walk(node):
if not isinstance(node, Element):
return
rect = node.getBoundingClientRect()
if rect.left <= x <= rect.right and rect.top <= y <= rect.bottom:
matches.append(node)
for child in getattr(node, "childNodes", []):
walk(child)
walk(self)
return matches
[docs]
def caretPositionFromPoint(self, x: float, y: float) -> CaretPosition | None:
"""Returns a CaretPosition for the closest element within this subtree."""
target = self.elementFromPoint(x, y)
if target is None:
return None
first_child = target.firstChild
if isinstance(first_child, Text):
return CaretPosition(first_child, 0)
return CaretPosition(target, 0)
[docs]
def getElementsByTagName(self, tagName: str) -> "HTMLCollection":
"""[Returns a collection of all child elements with the specified tag name
Args:
tagName (str): [a DOMString representing the tag name to match]
Returns:
[type]: [method returns a live HTMLCollection of elements with the given tag name.]
"""
elements = HTMLCollection()
def anon(el):
if self._matchElement(el, tagName):
elements.append(el)
self._iterate(self, anon)
return elements
[docs]
def hasAttribute(self, attribute: str) -> bool:
"""Returns True if an element has the specified attribute, otherwise False
Args:
attribute (str): [the attribute to test for]
Returns:
bool: [True if an element has the specified attribute, otherwise False]
"""
try:
if attribute[0:1] != "_":
attribute = "_" + attribute
return attribute in self.kwargs.keys()
except AttributeError:
return False
[docs]
def hasAttributes(self) -> bool:
"""Returns true if an element has any attributes, otherwise false"""
if len(self.kwargs) > 0:
return True
else:
return False
@property
def id(self) -> str | None:
"""Sets or returns the value of the id attribute of an element"""
return self.getAttribute("id")
@id.setter
def id(self, newid: str):
"""Sets or returns the value of the id attribute of an element"""
self.setAttribute("id", newid)
# Sets or returns the text content of a node and its descendants
def innerText(self, *args: Any) -> str:
if args:
self.textContent = "".join(str(each) for each in args)
return self.textContent
return self.textContent
# Inserts an element adjacent to the current element
def _normalize_adjacent_position(self, position: str) -> str:
pos = str(position).lower()
if pos not in ("beforebegin", "afterbegin", "beforeend", "afterend"):
raise ValueError(
f"The value provided ({position}) is not one of"
'"beforeBegin", "afterBegin", "beforeEnd", or "afterEnd".'
)
return pos
def _coerce_adjacent_nodes(self, content: Any) -> list[Any]:
if isinstance(content, tuple):
return list(content)
if isinstance(content, list):
return content
return [content]
def before(self, *nodes: Any) -> None:
if self.parentNode is None:
return
parent = self.parentNode
index = parent.args.index(self)
parent.args = parent.args[:index] + tuple(nodes) + parent.args[index:]
parent._update_parents()
def after(self, *nodes: Any) -> None:
if self.parentNode is None:
return
parent = self.parentNode
index = parent.args.index(self) + 1
parent.args = parent.args[:index] + tuple(nodes) + parent.args[index:]
parent._update_parents()
[docs]
def insertAdjacentElement(self, position: str, element: Element) -> Element | None:
"""Inserts an element adjacent to the current element."""
pos = self._normalize_adjacent_position(position)
if pos == "beforebegin":
if self.parentNode is None:
return None
self.before(element)
elif pos == "afterbegin":
self.prepend(element)
elif pos == "beforeend":
self.append(element)
elif pos == "afterend":
if self.parentNode is None:
return None
self.after(element)
return element
[docs]
def insertAdjacentHTML(self, position: str, html: str) -> None:
"""Inserts raw HTML adjacent to the current element"""
from domonic import domonic
try:
content = domonic.load(html)
except Exception:
content = html
nodes = self._coerce_adjacent_nodes(content)
pos = self._normalize_adjacent_position(position)
if pos == "beforebegin":
self.before(*nodes)
elif pos == "afterbegin":
self.prepend(*nodes)
elif pos == "beforeend":
self.append(*nodes)
elif pos == "afterend":
self.after(*nodes)
[docs]
def insertAdjacentText(self, position: str, text: str) -> None:
"""Inserts text adjacent to the current element"""
pos = self._normalize_adjacent_position(position)
if pos == "beforebegin":
self.before(text)
elif pos == "afterbegin":
self.prepend(text)
elif pos == "beforeend":
self.append(text)
elif pos == "afterend":
self.after(text)
[docs]
def isContentEditable(self) -> bool:
"""Returns true if the content of an element is editable, otherwise false"""
if self.getAttribute("contenteditable") == "true":
return True
return False
# def lang(self) -> str:
# """ Sets or returns the value of the lang attribute of an element """ # TODO - prop?
# return self.getAttribute('lang')
[docs]
def lastElementChild(self) -> Node | None:
"""[Returns the last child element of an element]
Returns:
[type]: [the last child element of an element]
"""
try:
return self.args[len(self.args) - 1]
except Exception:
return None
[docs]
def namespaceURI(self) -> str:
"""Returns the namespace URI of an element"""
return getattr(self, "_namespaceURI", "http://www.w3.org/1999/xhtml")
@property
def nextSibling(self) -> Node | None:
"""Returns the next node at the same node tree level"""
if self.parentNode is not None:
for count, el in enumerate(self.parentNode.args):
if el is self and count < len(self.parentNode.args) - 1:
return self.parentNode.args[count + 1]
return None
@property
def nextElementSibling(self) -> Node | None:
"""Returns the next element at the same node tree level"""
if self.parentNode is not None:
for count, el in enumerate(self.parentNode.args):
if el is self and count < len(self.parentNode.args) - 1:
if type(self.parentNode.args[count + 1]) is not str:
return self.parentNode.args[count + 1]
return None
@property
def previousElementSibling(self) -> Node | None:
"""returns the Element immediately prior to the specified one in its parent's children list,
or None if the specified element is the first one in the list."""
if self.parentNode is not None:
for count, el in enumerate(self.parentNode.args):
if el is self and count > 0:
if type(self.parentNode.args[count - 1]) is not str:
return self.parentNode.args[count - 1]
return None
[docs]
def normalize(self) -> list[Any]:
"""Joins adjacent text nodes and removes empty text nodes in an element"""
content = []
nodestr = ""
for s in self.args:
if type(s) == Text:
# content.append(s.textContent)
nodestr += s.textContent
continue
elif type(s) == str:
nodestr += s
continue
elif nodestr != "":
content.append(nodestr)
nodestr = ""
elif type(s) != str:
content.append(s)
if nodestr != "":
content.append(nodestr)
self.args = content
return self.args
[docs]
def offsetHeight(self) -> float:
"""Returns the height of an element, including padding, border and scrollbar"""
return (
self.clientHeight
+ Element._style_number(self.style.borderTopWidth)
+ Element._style_number(self.style.borderBottomWidth)
)
[docs]
def offsetWidth(self) -> float:
"""Returns the width of an element, including padding, border and scrollbar"""
return (
self.clientWidth
+ Element._style_number(self.style.borderLeftWidth)
+ Element._style_number(self.style.borderRightWidth)
)
[docs]
def offsetLeft(self) -> float:
"""Returns the horizontal offset position of an element"""
return Element._style_number(self.style.left)
[docs]
def offsetParent(self) -> Node | None:
"""Returns the offset container of an element"""
return self.parentNode
[docs]
def offsetTop(self) -> float:
"""Returns the vertical offset position of an element"""
return Element._style_number(self.style.top)
@property
def parentElement(self) -> Node | None:
"""Returns the parent element node of an element"""
return self.parentNode
# @property
# def previousSibling(self):
# """ Returns the previous node at the same node tree level """
# if self.parentNode is not None:
# for count, el in enumerate(self.parentNode.args):
# if el is self and count > 1:
# return self.parentNode.args[count - 1]
# return None
[docs]
def prepend(self, *args: Any) -> None:
"""Prepends a node to the current element"""
newargs = list(args) + list(self.args)
self.args = tuple(newargs)
self._update_parents()
[docs]
def querySelector(self, query: str) -> Element | None:
"""[Returns the first child element that matches a specified CSS selector(s) of an element]
Args:
query (str): [a CSS selector string]
Returns:
[type]: [an Element object]
"""
try:
return self.querySelectorAll(query)[0]
except Exception as e:
return None
[docs]
def querySelectorAll(self, query: str) -> list[Element]:
"""[Returns all child elements that matches a specified CSS selector(s) of an element]
Args:
query (str): [a CSS selector string]
Returns:
[type]: [a list of Element objects]
"""
if not query:
return []
query = query.strip()
def _fallback_selector_results():
if query.startswith("."):
return self.getElementsByClassName(query[1:])
if query.startswith("#"):
found = self.getElementById(query[1:])
return [found] if found else []
results = self.getElementsBySelector(query, self)
return results if isinstance(results, list) else []
naked_query = query[1:]
if "." in naked_query or "[" in naked_query or " " in naked_query:
try:
from cssselect import HTMLTranslator, SelectorError
expression = HTMLTranslator().css_to_xpath(query)
from domonic.webapi.xpath import XPathEvaluator, XPathResult
evaluator = XPathEvaluator()
expression = evaluator.createExpression(expression)
result = expression.evaluate(self, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE)
if result.nodes:
return result.nodes
return _fallback_selector_results()
except ImportError:
return _fallback_selector_results()
except SelectorError:
print("Invalid selector.")
return []
elements = []
def anon(el):
if self._matchElement(el, query):
elements.append(el)
self._iterate(self, anon)
return elements
[docs]
def remove(self):
"""Removes the element from the DOM"""
# try:
# self.parentNode.args.remove(self)
# except Exception:
# # print("Element not found")
# pass
# if self.parentNode is None:
# self._update_parents()
if self.parentNode is not None:
self.parentNode.removeChild(self)
return self
[docs]
def removeAttribute(self, attribute: str):
"""Removes a specified attribute from an element"""
if attribute[0:1] != "_":
attribute = "_" + attribute
if attribute not in self.kwargs:
return None
old_value = self.kwargs.get(attribute)
del self.kwargs[attribute]
_notify_attribute_changed(self, attribute, old_value, None)
_queue_mutation_record(
"attributes",
self,
attribute_name=attribute[1:] if attribute.startswith("_") else attribute,
old_value=str(old_value) if old_value is not None else None,
)
return None
[docs]
def removeAttributeNode(self, attribute): # untested
"""Removes a specified attribute node, and returns the removed node"""
for each in self.kwargs:
if attribute == each:
val = self.kwargs[each]
del self.kwargs[each]
_notify_attribute_changed(self, attribute, val, None)
_queue_mutation_record(
"attributes",
self,
attribute_name=attribute[1:] if isinstance(attribute, str) and attribute.startswith("_") else attribute,
old_value=str(val) if val is not None else None,
)
return Attr(attribute, val)
[docs]
def requestFullscreen(self):
"""Shows an element in fullscreen mode"""
doc = self.ownerDocument
if doc is not None:
doc._fullscreenElement = self
return self
# def setPointerCapture(self):
# ''' Sets the pointer capture to the specified element '''
# raise NotImplementedError
# def releasePointerCapture(self):
# ''' Releases the pointer capture from the specified element '''
# raise NotImplementedError
[docs]
def setAttribute(self, attribute, value):
"""Sets or changes the specified attribute, to the specified value"""
try:
if attribute[0:1] != "_":
attribute = "_" + attribute
old_value = self.kwargs.get(attribute)
self.kwargs[attribute] = value
_notify_attribute_changed(self, attribute, old_value, value)
_queue_mutation_record(
"attributes",
self,
attribute_name=attribute[1:] if attribute.startswith("_") else attribute,
old_value=str(old_value) if old_value is not None else None,
)
except Exception as e:
# print('failed to set attribute', e)
return None
[docs]
def setAttributeNode(self, attr):
"""[Sets or changes the specified attribute node]
Args:
attr ([type]): [an Attr object]
"""
self.setAttribute(attr.name, attr.value)
@property
def style(self):
"""returns the value of the style attribute of an element"""
if self.__style is None:
self.style = Style()
return self.__style
@style.setter
def style(self, style):
self.__style = style
self.__style.__init__(self) # to set the parent
# def tabIndex(self):
# ''' Sets or returns the value of the tabindex attribute of an element'''
# pass
@property
def tagName(self):
return self.name
# @property
# def textContent(self):
# return self.nodeValue
# @textContent.setter
# def textContent(self, content):
# self.nodeValue = content
@property
def title(self):
"""returns the value of the title attribute of an element"""
return self.getAttribute("title")
@title.setter
def title(self, newtitle: str):
"""[Sets the value of the title attribute of an element]
Args:
newtitle (str): [the new title value]
"""
self.setAttribute("title", newtitle)
[docs]
def toString(self) -> str:
"""Converts an element to a string"""
return str(self)
class DOMImplementation:
def __init__(self):
# self.__domImplementation = None
pass
def createDocument(self, namespaceURI: str, qualifiedName: str, doctype: str):
if namespaceURI is None:
namespaceURI = ""
if qualifiedName is None:
qualifiedName = ""
if doctype is None:
doctype = ""
d = XMLDocument()
root = d.createElementNS(namespaceURI, qualifiedName) if qualifiedName else None
if root is not None:
d.args = (root,)
root.parentNode = d
d.documentElement = root
d.doctype = doctype
return d
def createDocumentType(self, qualifiedName: str, publicId: str, systemId: str) -> DocumentType:
"""[creates a DocumentType node]
Args:
qualifiedName (str): [the qualified name of the document type]
publicId (str): [the public identifier of the document type]
systemId (str): [the system identifier of the document type]
Returns:
[type]: [a DocumentType object]
"""
return DocumentType(qualifiedName, publicId, systemId)
def createHTMLDocument(self, title=None):
doc = HTMLDocument()
html_el = Document.createElement("html")
head_el = Document.createElement("head")
body_el = Document.createElement("body")
if title is not None:
head_el.appendChild(Document.createElement("title", title))
html_el.appendChild(head_el)
html_el.appendChild(body_el)
doc.args = (html_el,)
html_el.parentNode = doc
doc.documentElement = html_el
return doc
def hasFeatures(self, featureList) -> bool:
return True
[docs]
class ProcessingInstruction(Node):
nodeType: int = Node.PROCESSING_INSTRUCTION_NODE
__slots__ = ("target", "data")
def __init__(self, target, data) -> None:
super().__init__()
self.target = target
self.data = data
def toString(self) -> str:
return f"<?{self.target} {self.data}?>"
__str__ = toString
[docs]
class CDATASection(Node):
"""The CDATASection interface represents a CDATA section that can be used within XML
to include extended portions of unescaped text, such that the symbols < and & do not
need escaping as they normally do within XML when used as text."""
nodeType: int = Node.CDATA_SECTION_NODE
__slots__ = "data"
def __init__(self, data) -> None:
self.data = data
def toString(self) -> str:
return f"<![CDATA[{self.data}]]>"
__str__ = toString
@property
def __len__(self) -> int:
return len(self.data)
@property
def length(self) -> int:
return len(self.data)
# def __format__(self, format_spec):
# return str(self)
class AbastractRange:
def __init__(self) -> None:
"""Constructor for Range objects"""
self.startContainer: Node | None = None
self.startOffset: int = 0
self.endContainer: Node | None = None
self.endOffset: int = 0
self.collapsed: bool = True
self.commonAncestorContainer: Node | None = None
def cloneContents(self) -> "DocumentFragment":
return self.cloneRange().cloneContents()
def cloneRange(self) -> "Range":
new_range = Range()
if self.startContainer is not None:
new_range.setStart(self.startContainer, self.startOffset)
if self.endContainer is not None:
new_range.setEnd(self.endContainer, self.endOffset)
return new_range
def compareBoundaryPoints(self, how: int, sourceRange: "Range") -> int:
return self.cloneRange().compareBoundaryPoints(how, sourceRange)
def createContextualFragment(self, data: Any) -> "DocumentFragment":
return self.cloneRange().createContextualFragment(data)
def deleteContents(self) -> None:
self.cloneRange().deleteContents()
def detach(self) -> None:
self.startContainer = None
self.endContainer = None
self.startOffset = 0
self.endOffset = 0
self.collapsed = True
self.commonAncestorContainer = None
def expand(self, unit: Any) -> None:
if self.startContainer is None:
return
unit = str(unit or "").lower()
if unit in ("all", "document", "container"):
self.selectNodeContents(self.commonAncestorContainer or self.startContainer)
return
if not isinstance(self.startContainer, Text) or self.startContainer is not self.endContainer:
self.selectNodeContents(self.commonAncestorContainer or self.startContainer)
return
text = self.startContainer.textContent
if unit == "character":
start = max(0, min(self.startOffset, len(text)))
end = max(start, min(len(text), start + 1))
self.setStart(self.startContainer, start)
self.setEnd(self.startContainer, end)
return
if unit == "word":
start = max(0, min(self.startOffset, len(text)))
end = max(start, min(self.endOffset, len(text)))
while start > 0 and not text[start - 1].isspace():
start -= 1
while end < len(text) and not text[end].isspace():
end += 1
self.setStart(self.startContainer, start)
self.setEnd(self.startContainer, end)
return
self.selectNodeContents(self.startContainer)
def extractContents(self) -> "DocumentFragment":
return self.cloneRange().extractContents()
def getBoundingClientRect(self) -> DOMRect:
return self.cloneRange().getBoundingClientRect()
def getClientRects(self) -> DOMRectList:
return self.cloneRange().getClientRects()
def insertNode(self, newNode: "Node") -> None:
self.cloneRange().insertNode(newNode)
def selectNode(self, refNode: "Node") -> None:
if refNode.parentNode is None:
raise ValueError("Cannot select a detached node")
self.setStartBefore(refNode)
self.setEndAfter(refNode)
def selectNodeContents(self, refNode: "Node") -> None:
self.setStart(refNode, 0)
self.setEnd(refNode, Range._container_length(refNode))
def setEnd(self, refNode: "Node", offset: int) -> None:
self.endContainer = refNode
self.endOffset = offset
if self.startContainer is None:
self.startContainer = refNode
self.startOffset = offset
self._update_state()
def setEndAfter(self, refNode: "Node") -> None:
self.setEnd(refNode.parentNode, list(refNode.parentNode.childNodes).index(refNode) + 1)
def setEndBefore(self, refNode: "Node") -> None:
self.setEnd(refNode.parentNode, list(refNode.parentNode.childNodes).index(refNode))
def setStart(self, refNode: "Node", offset: int) -> None:
self.startContainer = refNode
self.startOffset = offset
if self.endContainer is None:
self.endContainer = refNode
self.endOffset = offset
self._update_state()
def setStartAfter(self, refNode: "Node") -> None:
self.setStart(refNode.parentNode, list(refNode.parentNode.childNodes).index(refNode) + 1)
def setStartBefore(self, refNode: "Node") -> None:
self.setStart(refNode.parentNode, list(refNode.parentNode.childNodes).index(refNode))
def surroundContents(self, newParent: "Node") -> None:
self.cloneRange().surroundContents(newParent)
def toString(self) -> str:
return self.cloneRange().toString()
def comparePoint(self, refNode: "Node", offset: int) -> int:
return self.cloneRange().comparePoint(refNode, offset)
def deleteData(self, offset, count):
if not isinstance(self.startContainer, Text) or self.startContainer is not self.endContainer:
raise ValueError("Range data helpers require a single Text container")
self.startContainer.deleteData(offset, count)
self.endOffset = min(self.endOffset, len(self.startContainer.textContent))
self.startOffset = min(self.startOffset, self.endOffset)
self._update_state()
return self.startContainer.textContent
def extractData(self, offset, count):
data = self.getData(offset, count)
self.deleteData(offset, count)
return data
def getData(self, offset, count):
if not isinstance(self.startContainer, Text) or self.startContainer is not self.endContainer:
raise ValueError("Range data helpers require a single Text container")
return self.startContainer.textContent[offset : offset + count]
def getEnd(self):
return (self.endContainer, self.endOffset)
def getStart(self):
return (self.startContainer, self.startOffset)
def replaceData(self, offset, count, data):
if not isinstance(self.startContainer, Text) or self.startContainer is not self.endContainer:
raise ValueError("Range data helpers require a single Text container")
self.startContainer.replaceData(offset, count, data)
self.endOffset = min(len(self.startContainer.textContent), max(self.startOffset, self.endOffset))
self._update_state()
return self.startContainer.textContent
def setData(self, data):
if not isinstance(self.startContainer, Text) or self.startContainer is not self.endContainer:
raise ValueError("Range data helpers require a single Text container")
self.startContainer.data = data
self.startOffset = min(self.startOffset, len(data))
self.endOffset = min(self.endOffset, len(data))
self._update_state()
return self.startContainer.textContent
def _update_state(self) -> None:
self.collapsed = (
self.startContainer is self.endContainer and self.startOffset == self.endOffset
)
if self.startContainer is None or self.endContainer is None:
self.commonAncestorContainer = None
return
start_path = Range._path_to_root(self.startContainer)
end_path = Range._path_to_root(self.endContainer)
ancestor = None
while start_path and end_path and start_path[-1] is end_path[-1]:
ancestor = start_path.pop()
end_path.pop()
self.commonAncestorContainer = ancestor
AbstractRange = AbastractRange
class Range(AbastractRange):
# TODO - untested
START_TO_START: ClassVar[int] = 0
START_TO_END: ClassVar[int] = 1
END_TO_END: ClassVar[int] = 2
END_TO_START: ClassVar[int] = 3
def __init__(self) -> None:
self.startContainer: Node | None = None
self.startOffset = 0
self.endContainer: Node | None = None
self.endOffset = 0
self.collapsed = True
self.commonAncestorContainer: Node | None = None
@staticmethod
def _container_length(node: Node | None) -> int:
if node is None:
return 0
if isinstance(node, Text):
return len(node.textContent)
return len(getattr(node, "childNodes", []))
@staticmethod
def _path_to_root(node: Node | None) -> list[Node]:
path: list[Node] = []
current = node
while current is not None:
path.append(current)
current = getattr(current, "parentNode", None)
return path
@staticmethod
def _compare_points(node_a: Node, offset_a: int, node_b: Node, offset_b: int) -> int:
if node_a is node_b:
if offset_a < offset_b:
return -1
if offset_a > offset_b:
return 1
return 0
path_a = Range._path_to_root(node_a)
path_b = Range._path_to_root(node_b)
common = None
while path_a and path_b and path_a[-1] is path_b[-1]:
common = path_a.pop()
path_b.pop()
if common is None:
return 0
child_a = path_a[-1] if path_a else common
child_b = path_b[-1] if path_b else common
siblings = list(getattr(common, "childNodes", []))
try:
index_a = siblings.index(child_a)
index_b = siblings.index(child_b)
except ValueError:
return 0
if index_a < index_b:
return -1
if index_a > index_b:
return 1
return 0
def _update_state(self) -> None:
self.collapsed = (
self.startContainer is self.endContainer and self.startOffset == self.endOffset
)
if self.startContainer is None or self.endContainer is None:
self.commonAncestorContainer = None
return
start_path = self._path_to_root(self.startContainer)
end_path = self._path_to_root(self.endContainer)
ancestor = None
while start_path and end_path and start_path[-1] is end_path[-1]:
ancestor = start_path.pop()
end_path.pop()
self.commonAncestorContainer = ancestor
def _common_ancestor_child_slice(self) -> tuple[Node, int, int] | None:
ancestor = self.commonAncestorContainer
if ancestor is None or isinstance(ancestor, Text):
return None
children = list(getattr(ancestor, "childNodes", []))
def resolve_index(node: Node | None, offset: int, *, is_end: bool) -> int | None:
if node is None:
return None
if node is ancestor:
bounded = max(0, min(offset, len(children)))
return bounded
current = node
while current is not None and getattr(current, "parentNode", None) is not ancestor:
current = getattr(current, "parentNode", None)
if current is None:
return None
try:
index = children.index(current)
except ValueError:
return None
if is_end:
if isinstance(node, Text) and offset == 0:
return index
return index + 1
return index
start_index = resolve_index(self.startContainer, self.startOffset, is_end=False)
end_index = resolve_index(self.endContainer, self.endOffset, is_end=True)
if start_index is None or end_index is None:
return None
return ancestor, start_index, end_index
@classmethod
def _validate_boundary_point(cls, node: Node, offset: int) -> int:
if node is None:
raise ValueError("Boundary node cannot be None")
if not isinstance(offset, int):
raise TypeError("Range offset must be an integer")
max_offset = cls._container_length(node)
if offset < 0 or offset > max_offset:
raise ValueError("Range offset is out of bounds")
return offset
def setStart(self, node: Node, offset: int) -> None:
offset = self._validate_boundary_point(node, offset)
self.startContainer = node
self.startOffset = offset
if self.endContainer is None:
self.endContainer = node
self.endOffset = offset
elif self._compare_points(node, offset, self.endContainer, self.endOffset) > 0:
self.endContainer = node
self.endOffset = offset
self._update_state()
def setEnd(self, node: Node, offset: int) -> None:
offset = self._validate_boundary_point(node, offset)
self.endContainer = node
self.endOffset = offset
if self.startContainer is None:
self.startContainer = node
self.startOffset = offset
elif self._compare_points(node, offset, self.startContainer, self.startOffset) < 0:
self.startContainer = node
self.startOffset = offset
self._update_state()
def setStartBefore(self, node: Node) -> None:
self.setStart(node.parentNode, list(node.parentNode.childNodes).index(node))
def setStartAfter(self, node: Node) -> None:
self.setStart(node.parentNode, list(node.parentNode.childNodes).index(node) + 1)
def setEndBefore(self, node: Node) -> None:
self.setEnd(node.parentNode, list(node.parentNode.childNodes).index(node))
def setEndAfter(self, node: Node) -> None:
self.setEnd(node.parentNode, list(node.parentNode.childNodes).index(node) + 1)
def collapse(self, toStart: bool) -> None:
if toStart:
self.endContainer = self.startContainer
self.endOffset = self.startOffset
else:
self.startContainer = self.endContainer
self.startOffset = self.endOffset
self._update_state()
def selectNode(self, node: Node) -> None:
self.setStartBefore(node)
self.setEndAfter(node)
def selectNodeContents(self, node: Node) -> None:
self.setStart(node, 0)
self.setEnd(node, self._container_length(node))
def compareBoundaryPoints(self, how: int, sourceRange: "Range") -> int:
comparisons = {
self.START_TO_START: (self.startContainer, self.startOffset, sourceRange.startContainer, sourceRange.startOffset),
self.START_TO_END: (self.startContainer, self.startOffset, sourceRange.endContainer, sourceRange.endOffset),
self.END_TO_END: (self.endContainer, self.endOffset, sourceRange.endContainer, sourceRange.endOffset),
self.END_TO_START: (self.endContainer, self.endOffset, sourceRange.startContainer, sourceRange.startOffset),
}
if how not in comparisons:
raise ValueError("Invalid Range comparison type")
return self._compare_points(*comparisons[how])
def deleteContents(self) -> None:
self.extractContents()
def extractContents(self) -> "DocumentFragment":
if self.startContainer is None:
return DocumentFragment()
if isinstance(self.startContainer, Text) and self.startContainer == self.endContainer:
text = self.startContainer.textContent
extracted = text[self.startOffset : self.endOffset]
self.startContainer.textContent = text[: self.startOffset] + text[self.endOffset :]
self.endContainer = self.startContainer
self.endOffset = self.startOffset
self._update_state()
return DocumentFragment(Text(extracted))
if self.startContainer == self.endContainer:
container = self.startContainer
children = list(container.childNodes)
extracted = children[self.startOffset : self.endOffset]
if hasattr(container, "args"):
kept = children[: self.startOffset] + children[self.endOffset :]
container.args = tuple(kept)
for child in kept:
if isinstance(child, Node):
child.parentNode = container
self.endContainer = container
self.endOffset = self.startOffset
self._update_state()
return DocumentFragment(*extracted)
child_slice = self._common_ancestor_child_slice()
if child_slice is not None:
container, start_index, end_index = child_slice
children = list(container.childNodes)
extracted = children[start_index:end_index]
if hasattr(container, "args"):
kept = children[:start_index] + children[end_index:]
container.args = tuple(kept)
for child in kept:
if isinstance(child, Node):
child.parentNode = container
self.startContainer = container
self.endContainer = container
self.startOffset = start_index
self.endOffset = start_index
self._update_state()
return DocumentFragment(*extracted)
return DocumentFragment()
def cloneContents(self) -> "DocumentFragment":
import copy
if self.startContainer is None:
return DocumentFragment()
if isinstance(self.startContainer, Text) and self.startContainer == self.endContainer:
return DocumentFragment(Text(self.startContainer.textContent[self.startOffset : self.endOffset]))
if self.startContainer == self.endContainer:
container = self.startContainer
children = list(container.childNodes)
cloned = [copy.deepcopy(child) for child in children[self.startOffset : self.endOffset]]
return DocumentFragment(*cloned)
child_slice = self._common_ancestor_child_slice()
if child_slice is not None:
container, start_index, end_index = child_slice
children = list(container.childNodes)
cloned = [copy.deepcopy(child) for child in children[start_index:end_index]]
return DocumentFragment(*cloned)
return DocumentFragment()
def getBoundingClientRect(self) -> DOMRect:
rects = self.getClientRects()
if not rects:
return DOMRect(0, 0, 0, 0)
left = min(rect.left for rect in rects)
top = min(rect.top for rect in rects)
right = max(rect.right for rect in rects)
bottom = max(rect.bottom for rect in rects)
return DOMRect(left, top, right - left, bottom - top)
def getClientRects(self) -> DOMRectList:
if self.startContainer is None:
return DOMRectList()
if isinstance(self.startContainer, Text) and self.startContainer == self.endContainer:
parent = getattr(self.startContainer, "parentNode", None)
return DOMRectList([parent.getBoundingClientRect()]) if hasattr(parent, "getBoundingClientRect") else DOMRectList()
if self.startContainer == self.endContainer:
rects = []
for child in list(self.startContainer.childNodes)[self.startOffset : self.endOffset]:
if hasattr(child, "getBoundingClientRect"):
rects.append(child.getBoundingClientRect())
return DOMRectList(rects)
child_slice = self._common_ancestor_child_slice()
if child_slice is not None:
container, start_index, end_index = child_slice
rects = []
for child in list(container.childNodes)[start_index:end_index]:
if hasattr(child, "getBoundingClientRect"):
rects.append(child.getBoundingClientRect())
return DOMRectList(rects)
return DOMRectList()
def insertNode(self, node: Node) -> None:
if self.startContainer is None:
return
container = self.startContainer
if isinstance(container, Text):
text = container.textContent
before = Text(text[: self.startOffset])
after = Text(text[self.startOffset :])
parent = container.parentNode
if parent is None:
return
children = list(parent.childNodes)
index = children.index(container)
replacement = [part for part in (before, node, after) if part.textContent != "" if isinstance(part, Text)] if False else None
new_children = children[:index]
if before.textContent != "":
new_children.append(before)
before.parentNode = parent
new_children.append(node)
node.parentNode = parent
if after.textContent != "":
new_children.append(after)
after.parentNode = parent
new_children.extend(children[index + 1 :])
parent.args = tuple(new_children)
self.startContainer = parent
self.endContainer = parent
self.startOffset = index + 1
self.endOffset = index + 1
self._update_state()
return
if hasattr(container, "insertBefore"):
children = list(container.childNodes)
ref = children[self.startOffset] if self.startOffset < len(children) else None
container.insertBefore(node, ref)
self.startOffset += 1
self.endOffset = max(self.endOffset, self.startOffset)
self._update_state()
def surroundContents(self, newParent: Node) -> None:
fragment = self.extractContents()
for child in fragment.args:
newParent.appendChild(child)
self.insertNode(newParent)
self.selectNode(newParent)
def cloneRange(self) -> "Range":
new_range = Range()
new_range.startContainer = self.startContainer
new_range.startOffset = self.startOffset
new_range.endContainer = self.endContainer
new_range.endOffset = self.endOffset
new_range.collapsed = self.collapsed
new_range.commonAncestorContainer = self.commonAncestorContainer
return new_range
def detach(self) -> None:
self.startContainer = None
self.endContainer = None
self.startOffset = 0
self.endOffset = 0
self._update_state()
def createContextualFragment(self, fragment: Any) -> "DocumentFragment":
if isinstance(fragment, DocumentFragment):
return fragment
if isinstance(fragment, Node):
return DocumentFragment(fragment)
if not isinstance(fragment, str):
return DocumentFragment(fragment)
try:
from domonic import domonic
page = domonic.parseString(f"<body>{fragment}</body>")
if page is not None:
body = page.querySelector("body")
if body is not None:
return DocumentFragment(*list(body.childNodes))
except Exception:
pass
return DocumentFragment(fragment)
def toString(self) -> str:
if self.startContainer is None:
return ""
if isinstance(self.startContainer, Text) and self.startContainer == self.endContainer:
return self.startContainer.textContent[self.startOffset : self.endOffset]
if self.startContainer == self.endContainer:
container = self.startContainer
children = list(container.childNodes)
return "".join(str(child) for child in children[self.startOffset : self.endOffset])
child_slice = self._common_ancestor_child_slice()
if child_slice is not None:
container, start_index, end_index = child_slice
children = list(container.childNodes)
return "".join(str(child) for child in children[start_index:end_index])
return ""
def comparePoint(self, refNode: Node, offset: int) -> int:
if self.startContainer is None or self.endContainer is None:
raise Exception("Range has no boundaries")
offset = self._validate_boundary_point(refNode, offset)
if self._compare_points(refNode, offset, self.startContainer, self.startOffset) < 0:
return -1
if self._compare_points(refNode, offset, self.endContainer, self.endOffset) > 0:
return 1
return 0
def isPointInRange(self, refNode: Node, offset: int) -> bool:
return self.comparePoint(refNode, offset) == 0
def intersectsNode(self, refNode: Node) -> bool:
if self.startContainer is None or self.endContainer is None:
return False
if isinstance(refNode, Text):
start_node, start_offset = refNode, 0
end_node, end_offset = refNode, len(refNode.textContent)
else:
parent = getattr(refNode, "parentNode", None)
if parent is None:
return refNode is self.commonAncestorContainer
siblings = list(getattr(parent, "childNodes", []))
if refNode not in siblings:
return False
index = siblings.index(refNode)
start_node, start_offset = parent, index
end_node, end_offset = parent, index + 1
return not (
self._compare_points(end_node, end_offset, self.startContainer, self.startOffset) <= 0
or self._compare_points(start_node, start_offset, self.endContainer, self.endOffset) >= 0
)
[docs]
class StaticRange(Range):
"""Immutable snapshot of a range boundary pair.
``StaticRange`` mirrors the platform idea of a range-like object that can
be inspected and cloned back into a mutable ``Range`` but cannot be edited
in place.
"""
def __init__(self, startContainer, startOffset, endContainer, endOffset):
super().__init__()
self.startContainer = startContainer
self.startOffset = startOffset
self.endContainer = endContainer
self.endOffset = endOffset
self._update_state()
def _immutable(self, *args, **kwargs):
raise TypeError("StaticRange is immutable")
collapse = _immutable
createContextualFragment = Range.createContextualFragment
deleteContents = _immutable
deleteData = _immutable
detach = _immutable
expand = _immutable
extractContents = _immutable
insertNode = _immutable
replaceData = _immutable
selectNode = _immutable
selectNodeContents = _immutable
setData = _immutable
setEnd = _immutable
setEndAfter = _immutable
setEndBefore = _immutable
setStart = _immutable
setStartAfter = _immutable
setStartBefore = _immutable
surroundContents = _immutable
def toRange(self) -> Range:
return self.cloneRange()
class TimeRanges:
def __init__(self, *ranges):
self._ranges = []
for item in ranges:
if isinstance(item, (list, tuple)) and len(item) == 2:
self._ranges.append((item[0], item[1]))
self.length = len(self._ranges)
def start(self, index):
return self._ranges[index][0]
def end(self, index):
return self._ranges[index][1]
def __len__(self):
return self.length
[docs]
class Document(Element):
"""The Document interface represents the entire HTML or XML document."""
URL: ClassVar[URL | None] = None
def __init__(self, *args, **kwargs):
"""Constructor for Document objects"""
self.args = args
self.kwargs = kwargs
# self.documentURI = uri
# self.documentElement = self
self._open_filename = None
self._activeElement = None
self._defaultView = None
self._designMode = "off"
self._currentScript = None
self._cookie_store: dict[str, str] = {}
self._lastModified = formatdate(time.time(), usegmt=True)
self._referrer = ""
self._timeline = None
self.stylesheets = None
self.doctype = None
super().__init__(*args, **kwargs)
try:
global document
document = self
except Exception as e:
print("failed to set document", e)
def __new__(cls, *args, **kwargs):
instance = super().__new__(cls)
instance.__init__(*args, **kwargs)
instance.documentElement = instance
instance.URL = ""
instance.baseURI = ""
try:
global document
document = instance
except Exception as e:
print("failed to set document", e)
return instance
# TODO - still not great as it also returns 'links' when searching for 'li'
# @property
def _get_tags(self, tag): # TODO - still old
"""returns the tags you want"""
reg = f"(<{tag}.*?>.+?</{tag}>)"
closed_tags = [
"base",
"link",
"meta",
"hr",
"br",
"wbr",
"img",
"embed",
"param",
"source",
"track",
"area",
"col",
"input",
"keygen",
"command",
]
if tag in closed_tags:
reg = f"(<{tag}.*?/>)"
pattern = re.compile(reg)
tags = re.findall(pattern, str(self))
return tags
# def activeElement():
""" Returns the currently focused element in the document"""
# return
# def adoptNode(self, node):
# """ Adopts a node from another document """
# if node.ownerDocument is not None:
# node.ownerDocument.removeChild(node)
# node.ownerDocument = self
# return node
@property
def stylesheets(self):
if self.__stylesheets is None:
self.stylesheets = StyleSheetList()
self.stylesheets._populate_stylesheets_from_document(self)
return self.__stylesheets
@stylesheets.setter
def stylesheets(self, stylesheets):
self.__stylesheets = stylesheets
# self.__stylesheets.__init__(self) # to set the parent??
@property
def activeElement(self) -> Element | None:
"""Returns the currently focused element, or the body/document element fallback."""
if self._activeElement is not None:
return self._activeElement
return self.body or self.documentElement
@property
def timeline(self) -> DocumentTimeline:
if self._timeline is None:
self._timeline = DocumentTimeline(self)
return self._timeline
@property
def currentScript(self) -> Element | None:
if self._currentScript is not None:
return self._currentScript
scripts = self.scripts
return scripts[-1] if scripts else None
@currentScript.setter
def currentScript(self, script: Element | None) -> None:
self._currentScript = script
@property
def defaultView(self):
return self._defaultView
@defaultView.setter
def defaultView(self, value) -> None:
self._defaultView = value
@property
def designMode(self) -> str:
return self._designMode
@designMode.setter
def designMode(self, value: str) -> None:
normalized = str(value).strip().lower()
if normalized not in {"on", "off"}:
raise ValueError("designMode must be 'on' or 'off'")
self._designMode = normalized
[docs]
def hasFocus(self) -> bool:
"""Returns True when the document currently tracks a focused element."""
return self._activeElement is not None
@property
def anchors(self):
"""[get the anchors in the document]"""
# only the ones with a name
tags = self.querySelectorAll("a")
tags = [tag for tag in tags if tag.hasAttribute("name")]
return tags
@property
def applets(self):
"""Returns a collection of all <applet> elements in the document"""
return self.querySelectorAll("applet")
@property
def body(self):
"""Returns the <body> element in the document"""
return self.querySelector("body")
@body.setter
def body(self, el):
"""Sets the <body> element in the document"""
if not isinstance(el, HTMLBodyElement):
raise DOMException(
DOMException.TYPE_MISMATCH_ERR,
"The new body element is of type '" + str(type(el)) + "'. It must be a 'HTMLBodyElement'",
)
else:
if self.body is not None:
self.body.remove()
self += el
[docs]
def close(self):
"""Closes the output stream previously opened with document.open()"""
self._open_filename = None
@property
def cookie(self) -> str:
return "; ".join(f"{name}={value}" for name, value in self._cookie_store.items())
@cookie.setter
def cookie(self, value: str) -> None:
if value is None:
return
text = str(value).strip()
if not text or "=" not in text:
return
first_pair = text.split(";", 1)[0]
name, cookie_value = first_pair.split("=", 1)
self._cookie_store[name.strip()] = cookie_value.strip()
@property
def charset(self):
"""Returns the character encoding for the document. Deprecated: Use characterSet instead."""
return "UTF-8"
@property
def characterSet(self):
"""Returns the character encoding for the document"""
return "UTF-8"
[docs]
@staticmethod
def createAttribute(name: str) -> Attr:
"""Creates an attribute node"""
return Attr(name)
[docs]
@staticmethod
def createDocumentFragment(*args: Any) -> "DocumentFragment":
"""Creates an empty DocumentFragment node if not content passed. I added args as optional to pass content"""
return DocumentFragment(*args)
[docs]
@staticmethod
def createExpression(xpath: str, nsResolver: Any) -> XPathExpression:
"""Creates an XPathExpression object for the given XPath string."""
return XPathExpression(xpath, nsResolver)
[docs]
@staticmethod
def createElement(_type: str, *args: Any, **kwargs: Any) -> "Element":
"""Creates an Element node"""
from domonic.html import create_element
el = create_element(_type, *args, **kwargs)
if isinstance(el, Element):
_upgrade_custom_element_instance(el)
return el
[docs]
@staticmethod
def createElementNS(namespaceURI: str, qualifiedName: str, options: Any = None) -> "Element":
"""Creates an element with the specified namespace URI and qualified name."""
# el = type(qualifiedName, (Element,), {'name': qualifiedName})
from domonic.html import create_element
el = create_element(qualifiedName) # , *args, **kwargs)
el.namespaceURI = namespaceURI
_upgrade_custom_element_instance(el)
# el["name"] = qualifiedName
return el
[docs]
@staticmethod
def createEvent(event_type: str | None = None) -> Event:
"""[Creates a new event]
Args:
event_type ([type], optional): [description]. Defaults to None.
Returns:
[type]: [a new event]
"""
if event_type == "MouseEvent":
return MouseEvent("click")
if event_type == "PointerEvent":
from domonic.events import PointerEvent
return PointerEvent("pointerdown")
if event_type == "FocusEvent":
from domonic.events import FocusEvent
return FocusEvent("focus")
if event_type == "KeyboardEvent":
from domonic.events import KeyboardEvent
return KeyboardEvent("keydown")
if event_type == "UIEvent":
from domonic.events import UIEvent
return UIEvent("load")
if event_type == "CustomEvent":
from domonic.events import CustomEvent
return CustomEvent("custom")
if event_type == "CompositionEvent":
from domonic.events import CompositionEvent
return CompositionEvent("compositionstart")
if event_type == "SubmitEvent":
from domonic.events import SubmitEvent
return SubmitEvent("submit")
if event_type == "InputEvent":
from domonic.events import InputEvent
return InputEvent("input")
if event_type == "ClipboardEvent":
from domonic.events import ClipboardEvent
return ClipboardEvent("copy")
if event_type == "WheelEvent":
from domonic.events import WheelEvent
return WheelEvent("wheel")
if event_type == "BeforeUnloadEvent":
from domonic.events import BeforeUnloadEvent
return BeforeUnloadEvent("beforeunload")
if event_type == "MessageEvent":
from domonic.events import MessageEvent
return MessageEvent("message")
if event_type == "TransitionEvent":
from domonic.events import TransitionEvent
return TransitionEvent("transitionend")
if event_type == "ProgressEvent":
from domonic.events import ProgressEvent
return ProgressEvent("progress")
if event_type == "ErrorEvent":
from domonic.events import ErrorEvent
return ErrorEvent("error")
if event_type == "PopStateEvent":
from domonic.events import PopStateEvent
return PopStateEvent("popstate")
if event_type == "CloseEvent":
from domonic.events import CloseEvent
return CloseEvent("close")
if event_type is None:
return Event()
return Event(event_type)
[docs]
@staticmethod
def createTextNode(text: str) -> "Text":
"""[Creates a Text node with the specified text.
Args:
text ([str]): [the text to be inserted]
Returns:
[type]: [a new Text node]
"""
return Text(text)
[docs]
@staticmethod
def createTreeWalker(
root: Node, whatToShow: int | None = None, filter: Any = None, entityReferenceExpansion: Any = None
) -> "TreeWalker":
"""[creates a TreeWalker object]
Args:
root ([type]): [the root node at which to begin traversal]
whatToShow ([type], optional): [what types of nodes to show]. Defaults to None.
filter ([type], optional): [a NodeFilter or a function to be called for each node]. Defaults to None.
Returns:
[type]: [a new TreeWalker object]
"""
whatToShow = NodeFilter.SHOW_ALL if whatToShow == None else whatToShow
return TreeWalker(root, whatToShow, filter, entityReferenceExpansion)
[docs]
@staticmethod
def createProcessingInstruction(target: str, data: str) -> ProcessingInstruction:
"""Creates a ProcessingInstruction node with the specified target and data"""
return ProcessingInstruction(target, data)
[docs]
@staticmethod
def createEntityReference(name: str) -> "EntityReference":
"""Creates an EntityReference node with the specified name"""
return EntityReference(name)
@property
def xmlversion(self):
"""Returns the version of XML used for the document"""
return "1.0"
[docs]
@staticmethod
def createCDATASection(data: str) -> CDATASection:
"""Creates a CDATASection node with the specified data"""
return CDATASection(data)
# @staticmethod
# def createAttributeNS(namespaceURI, qualifiedName):
# """ Creates an Attr node with the specified namespace URI and qualified name """
# return Attr(qualifiedName)
[docs]
@staticmethod
def createRange() -> Range:
"""Creates a Range"""
return Range()
[docs]
@staticmethod
def createNodeIterator(root: Node, whatToShow: int | None = None, filter: Any = None) -> NodeIterator:
"""Creates a NodeIterator that can be used to traverse the document tree or subtree under root."""
whatToShow = NodeFilter.SHOW_ALL if whatToShow == None else whatToShow
return NodeIterator(root, whatToShow, filter)
# @staticmethod
# def caretRangeFromPoint(x, y):
# """ Returns the Range object that is the caret selection at the given coordinates. """
# return Range()
# raise NotImplementedError
# @staticmethod
# def createNSResolver(nodeResolver):
# """ Creates a NodeResolver """
# return NodeResolver(nodeResolver)
@property
def doctype(self):
"""Returns the Document Type Declaration associated with the document"""
return "<!DOCTYPE html>"
# return self.doctype = value
@doctype.setter
def doctype(self, value):
"""Sets the Document Type Declaration associated with the document"""
self._doctype = value
return
# def documentElement(self):
# ''' Returns the Document Element of the document (the <html> element)'''
# return self
# def documentMode(self):
""" Returns the mode used by the browser to render the document"""
# return
[docs]
def domain(self):
"""Returns the domain name of the server that loaded the document"""
try:
from domonic.webapi.url import URL
return URL(getattr(self, "URL", "")).hostname or ""
except Exception:
return ""
[docs]
def domConfig(self):
"""Returns the DOMConfig which has settings for how html content is rendered"""
return DOMConfig
[docs]
def elementFromPoint(self, x: float, y: float) -> Element | None:
"""Returns the topmost element at the specified coordinates."""
hits = self.elementsFromPoint(x, y)
return hits[0] if hits else None
[docs]
def evaluate(
self,
xpathExpression: str,
contextNode: "Node" = None,
namespaceResolver=None,
resultType=XPathResult.ORDERED_NODE_SNAPSHOT_TYPE,
result=None,
):
"""Evaluates an XPath expression and returns the result."""
if not isinstance(xpathExpression, str):
raise TypeError("xpathExpression must be a string")
if contextNode is None:
contextNode = self
evaluator = XPathEvaluator()
expression = evaluator.createExpression(xpathExpression)
result = expression.evaluate(contextNode, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE)
return result.nodes
[docs]
def elementsFromPoint(self, x: float, y: float) -> list[Element]:
"""Returns an array of all elements at the specified coordinates."""
matches = []
def walk(node):
if isinstance(node, Element):
rect = node.getBoundingClientRect()
if rect.left <= x <= rect.right and rect.top <= y <= rect.bottom:
matches.append(node)
for child in getattr(node, "childNodes", []):
if isinstance(child, str):
continue
walk(child)
walk(self)
return matches
[docs]
def caretPositionFromPoint(self, x: float, y: float) -> CaretPosition | None:
"""Returns a CaretPosition for the closest element at the given coordinates."""
target = self.elementFromPoint(x, y)
if target is None:
return None
first_child = target.firstChild
if isinstance(first_child, Text):
rect = target.getBoundingClientRect()
width = max(rect.width, 1)
text_length = len(first_child.textContent)
relative = max(0, min(x - rect.left, width))
offset = min(text_length, int((relative / width) * text_length))
return CaretPosition(first_child, offset)
return CaretPosition(target, 0)
@property
def embeds(self):
"""[Returns a collection of all <embed> elements the document]
Returns:
[type]: [a collection of all <embed> elements the document]
"""
return self.querySelectorAll("embed")
# def execCommand(self):
"""Invokes the specified clipboard operation on the element currently having focus."""
# return
@property
def forms(self):
"""Returns a collection of all <form> elements in the document"""
return self.querySelectorAll("form")
[docs]
def fullscreenElement(self):
"""Returns the current element that is displayed in fullscreen mode"""
return getattr(self, "_fullscreenElement", None)
[docs]
def fullscreenEnabled(self):
"""Returns a Boolean value indicating whether the document can be viewed in fullscreen mode"""
return False
[docs]
def getElementById(self, _id: str) -> Element | None:
"""[Returns the element that has the ID attribute with the specified value]
Args:
_id ([str]): [the value of the ID attribute]
Returns:
[type]: [the element that has the ID attribute with the specified value]
"""
for each in self.childNodes:
if not isinstance(each, Element):
continue
if each.getAttribute("id") == _id:
return each
for child in each.childNodes:
if not isinstance(child, Element):
continue
match = child._getElementById(_id)
# TODO - i think i need to build a hash map of IDs to positions on the tree
# for now I'm going to use recursion and add this same method to Element
if match is not False and match is not None:
return match
return False
[docs]
def getElementsByName(self, name: str):
"""[Returns a NodeList containing all elements with a specified name]
Args:
name (str): [the name to search for]
Returns:
[type]: [the matching elements]
"""
matches = HTMLCollection()
def walk(node):
if not isinstance(node, Element):
return
if node.getAttribute("name") == name:
matches.append(node)
for child in getattr(node, "childNodes", []):
walk(child)
for each in self.childNodes:
walk(each)
return matches
# def hasFocus():
# '''Returns a Boolean value indicating whether the document has focus'''
# return
@property
def head(self) -> "HTMLHeadElement":
"""Returns the <head> element of the document"""
return self.querySelector("head")
@head.setter
def head(self, el: "HTMLHeadElement") -> None:
"""[Sets the <head> element of the document]
Args:
el ([HTMLHeadElement]): [the new <head> element]
Raises:
DOMException: [if the el is not an HTMLHeadElement]
"""
if not isinstance(el, HTMLHeadElement):
raise DOMException("el must be an HTMLHeadElement")
self.removeChild(self.head)
if self.firstChild:
self.insertBefore(el, self.firstChild)
else:
self.appendChild(el)
@property
def images(self):
"""Returns a collection of all <img> elements in the document"""
return self.querySelectorAll("img")
@property
def implementation(self):
"""Returns the DOMImplementation object that handles this document"""
return DOMImplementation()
[docs]
def importNode(self, node, deep=False):
"""Imports a node from another document to this document."""
old_document = node.ownerDocument if isinstance(node, Node) else None
if isinstance(node, Element):
cloned = copy.deepcopy(node)
if not deep:
cloned.args = ()
cloned.ownerDocument = self
cloned._update_parents()
cloned._iterate(cloned, lambda current: setattr(current, "ownerDocument", self))
for current in _iter_dom_nodes(cloned):
if isinstance(current, Element):
_upgrade_custom_element_instance(current)
_run_adopted_callback(current, old_document, self)
return cloned
elif isinstance(node, Comment):
return Comment(node.data)
elif isinstance(node, Text):
return Text(node.data)
elif isinstance(node, ProcessingInstruction):
return ProcessingInstruction(node.target, node.data)
elif isinstance(node, DocumentFragment):
return DocumentFragment()
elif isinstance(node, Attr):
return Attr(node.name, node.value)
else:
raise Exception("Unsupported node type")
# def inputEncoding(self):
# """ Returns the encoding used to access the document's resources."""
# return
@property
def lastModified(self) -> str:
return self._lastModified
@lastModified.setter
def lastModified(self, value: str) -> None:
self._lastModified = str(value)
@property
def links(self):
"""Returns a collection of all <a> and <area> elements in the document that have a href attribute"""
anchors = [node for node in self.getElementsByTagName("a") if node.getAttribute("href") is not None]
areas = [node for node in self.getElementsByTagName("area") if node.getAttribute("href") is not None]
return anchors + areas
# @property
# def nodeType(self):
# return Node.DOCUMENT_NODE
nodeType: int = Node.DOCUMENT_NODE
[docs]
def normalizeDocument(self):
"""Removes empty Text nodes, and joins adjacent nodes"""
content = []
pending_text = ""
for each in list(self.childNodes):
if each.nodeType == Node.TEXT_NODE:
value = each.nodeValue or ""
if value.strip() == "":
continue
pending_text += value
continue
if pending_text != "":
content.append(Text(pending_text))
pending_text = ""
each.normalize()
content.append(each)
if pending_text != "":
content.append(Text(pending_text))
self.args = tuple(content)
self._update_parents()
return
[docs]
def open(self, index="index.html"):
"""Opens an HTML output stream to collect output from document.write()"""
self._open_filename = index
if not os.path.exists(index):
open(index, "w").close()
else:
print("File already exists")
# def readyState(self):
# ''' Returns the (loading) status of the document'''
# return
@property
def referrer(self) -> str:
return self._referrer
@referrer.setter
def referrer(self, value: str) -> None:
self._referrer = "" if value is None else str(value)
[docs]
def renameNode(self, node, namespaceURI: str, nodename: str):
"""[Renames the specified node, and returns the renamed node.]
Args:
node ([type]): [the node to rename]
namespaceURI ([type]): [a namespace URI]
nodename ([type]): [a node name]
Returns:
[type]: [description]
"""
if node.nodeType == Node.ELEMENT_NODE:
node.nodeName = nodename
node.namespaceURI = namespaceURI
return node
else:
return False
# def requestStorageAccess(self, storage_access_callback):
# """ Requests permission to access the user's storage area """
# return False
# def hasStorageAccess(self):
# """ Returns whether the user has granted permission to access the user's storage area """
# return False
# @property
# def pictureInPictureElement(self):
# """ Returns the element currently in Picture-in-Picture mode, if any. """
# return None
# def exitPictureInPicture(self):
# """ Exits Picture-in-Picture mode, if any. """
# return False
@property
def pictureInPictureEnabled(self):
"""Returns whether Picture-in-Picture mode is enabled."""
return False
@property
def scripts(self):
"""[Returns a collection of <script> elements in the document]
Returns:
[type]: [a collection of <script> elements in the document]
"""
return self.querySelectorAll("script")
[docs]
def strictErrorChecking(self):
"""Returns a Boolean value indicating whether to stop on the first error"""
return False
@property
def title(self) -> str:
"""[gets the title of the document]
Returns:
[str]: The title of the document
"""
if self.querySelector("title"):
return self.querySelector("title").textContent
return ""
@title.setter
def title(self, value: str):
"""[Sets the title of the document]
Args:
value ([str]): [the new title of the document]
"""
if self.querySelector("title"):
self.querySelector("title").textContent = value
else:
if not self.head:
self.head = HTMLHeadElement()
self.head.appendChild(HTMLTitleElement(value))
@property
def visibilityState(self):
"""Returns the visibility state of the document"""
return "visible"
[docs]
def write(self, html: str = ""):
"""[writes HTML text to a document
Args:
html (str, optional): [the content to write to the document]
"""
html = str(html)
current_open_filename = self._open_filename
if current_open_filename is not None:
# open the file and APPEND the html to the file without losing the previous content
with open(current_open_filename, "a") as f:
f.write(html)
content = DocumentFragment(html)
self.__init__(content)
self._open_filename = current_open_filename
[docs]
def writeln(self, html: str = ""):
"""[writes HTML text to a document, followed by a line break]
Args:
html (str, optional): [the content to write to the document]
"""
self.write(html + "\n")
# def __md__(self)
# def __rst__(self)
# def __json__(self)
class Location:
# TODO - move this to the window class and remove all domonic.javascript refs in this file
def __init__(self, url: str = None, *args, **kwargs) -> None:
self.href = url
def __str__(self) -> str:
return self.href
# def __repr__(self):
# return self.uri
def origin(self):
"""Returns the protocol, hostname and port number of a URL"""
from domonic.webapi.url import URL
return URL(self.href or "").origin
def search(self):
"""Sets or returns the querystring part of a URL"""
from domonic.webapi.url import URL
return URL(self.href or "").search
def assign(self, url: str = "") -> None:
"""Loads a new document"""
self.href = url
return None
def reload(self):
"""Reloads the current document"""
return self.href
def replace(self, url: str = ""):
"""Replaces the current document with a new one"""
self.href = url
return None
location = Location
[docs]
class DocumentFragment(Node):
nodeType: int = Node.DOCUMENT_FRAGMENT_NODE
def __init__(self, *args: Any) -> None:
super().__init__(*args)
querySelector = Document.querySelector
querySelectorAll = Document.querySelectorAll
getElementById = Document.getElementById
getElementsByTagName = Document.getElementsByTagName
_matchElement = Document._matchElement
attributes = Element.attributes
[docs]
def replaceChildren(self, newChildren: Iterable[Any]) -> None:
"""Replaces the childNodes of the DocumentFragment object."""
self.content.replaceChild(newChildren)
def __format__(self, format_spec):
return self.__str__()
def __str__(self) -> str:
return "".join([str(a) for a in self.args])
[docs]
class CharacterData(Node):
"""
The CharacterData abstract interface represents a Node object that contains characters.
This is an abstract interface, meaning there aren't any objects of type CharacterData:
it is implemented by other interfaces like Text, Comment, or ProcessingInstruction, which aren't abstract.
"""
nextElementSibling = Element.nextElementSibling
previousElementSibling = Element.previousElementSibling
remove = ChildNode.remove
replaceWith = ChildNode.replaceWith
before = ChildNode.before
after = ChildNode.after
[docs]
def appendData(self, data):
"""Appends the given DOMString to the CharacterData.data string; when this method returns,
data contains the concatenated DOMString."""
old_value = self.args[0]
updated = self.args[0] + data
self.args = (updated,)
_queue_mutation_record("characterData", self, old_value=old_value)
return updated
[docs]
def deleteData(self, offset: int, count: int):
"""Removes the specified amount of characters, starting at the specified offset,
from the CharacterData.data string; when this method returns, data contains the shortened DOMString."""
old_value = self.args[0]
updated = self.args[0][:offset] + self.args[0][offset + count :]
self.args = (updated,)
_queue_mutation_record("characterData", self, old_value=old_value)
return updated
[docs]
def insertData(self, offset: int, data):
"""Inserts the specified characters, at the specified offset, in the CharacterData.data string;
when this method returns, data contains the modified DOMString."""
old_value = self.args[0]
updated = self.args[0][:offset] + data + self.args[0][offset:]
self.args = (updated,)
_queue_mutation_record("characterData", self, old_value=old_value)
return updated
[docs]
def replaceData(self, offset: int, count: int, data):
"""Replaces the specified amount of characters, starting at the specified offset, with the specified DOMString;
when this method returns, data contains the modified DOMString."""
old_value = self.args[0]
updated = self.args[0][:offset] + data + self.args[0][offset + count :]
self.args = (updated,)
_queue_mutation_record("characterData", self, old_value=old_value)
return updated
# def replaceWith(self, newChildren):
# """ Replaces the characters in the children list of its parent with a set of Node or DOMString objects. """
# self.replaceChildren(newChildren) # parentNode?
[docs]
def substringData(self, offset: int, length: int):
"""Returns a DOMString containing the part of CharacterData.data of the specified length and
starting at the specified offset."""
return self.args[0][offset : offset + length]
[docs]
class EntityReference(Node):
"""
The EntityReference interface represents a reference to an entity, either parsed
or unparsed, in an Entity Node. Note that this is not a CharacterData node,
and does not have any child nodes.
"""
def __init__(self, *args) -> None:
self.args = args
def __str__(self) -> str:
return "".join([str(a) for a in self.args])
[docs]
@staticmethod
def ordinal(entityName: str):
"""Returns the character corresponding to the given entity name."""
if len(entityName) != 1:
raise ValueError("entityName must be a single character")
return ord(entityName)
[docs]
@staticmethod
def fromOrdinal(ordinal: int):
"""Returns the entity name corresponding to the given character."""
return chr(ordinal)
[docs]
class Entity(Node):
def __init__(self, *args) -> None:
self.args = args
def __str__(self) -> str:
return "".join([str(a) for a in self.args])
[docs]
@staticmethod
def fromName(entityName: str) -> str:
"""Returns the entity name corresponding to the given character."""
return chr(ord(entityName))
[docs]
@staticmethod
def fromChar(char: str) -> str:
"""Returns the character corresponding to the given entity name."""
return ord(char)
# class Notation(Node):
# def __init__(self, *args):
# self.args = args
# def __str__(self):
# return ''.join([str(a) for a in self.args])
# def getPublicId(self):
# """ Returns the public identifier of the notation. """
# return self.args[0]
# def getSystemId(self):
# """ Returns the system identifier of the notation. """
# return self.args[1]
[docs]
class Text(CharacterData):
"""Text Node"""
@property
def wholeText(self):
"""Returns a DOMString containing all the text content of the node and its descendants."""
if self.args and isinstance(self.args[0], str):
return self.args[0]
return ""
[docs]
def splitText(self, offset: int):
"""Splits the Text node into two Text nodes at the specified offset, keeping both in the tree as siblings.
The first node is returned, while the second node is discarded and exists outside the tree."""
current = self.args[0]
head = current[:offset]
tail = current[offset:]
self.args = (head,)
sibling = Text(tail)
sibling.parentNode = self.parentNode
if self.parentNode is not None and hasattr(self.parentNode, "args"):
siblings = list(self.parentNode.args)
try:
index = siblings.index(self)
siblings.insert(index + 1, sibling)
self.parentNode.args = tuple(siblings)
self.parentNode._update_parents()
except ValueError:
pass
return sibling
@property
def assignedSlot(self):
"""Returns the slot whose assignedNodes contains this node."""
return _assigned_slot_for_node(self)
@property
def data(self):
return self.args[0]
@data.setter
def data(self, data):
if not isinstance(data, str):
raise ValueError("Data must be a string.")
old_value = self.args[0] if self.args else ""
self.args = (data,)
_queue_mutation_record("characterData", self, old_value=old_value)
nodeType: int = Node.TEXT_NODE
@property
def nodeName(self):
return "#text"
@property
def childNodes(self):
return () # Text nodes have no children
@property
def firstChild(self):
return None
# @property
# def firstChild(self):
# return self.args[0]
# @property
# def textContent(self):
# return self.nodeValue
# @textContent.setter
# def textContent(self, content):
# self.nodeValue = content
def __str__(self) -> str:
return str(self.textContent)
def __format__(self, format_spec):
return str(self.textContent)
# def __repr__(self):
# return str(self.textContent)
def __iter__(self):
return iter(()) # No children for text nodes
[docs]
class HTMLCollection(list):
@property
def length(self) -> int:
return len(self)
def __str__(self) -> str:
return "".join([str(a) for a in self])
[docs]
def item(self, index: int) -> Node | None:
"""[gets the indexth item in the collection.
If index is greater than or equal to the number of nodes in the list, this returns null.]
Args:
index ([type]): [the index of the item to return.]
Returns:
[type]: [the node at the indexth position, or None]
"""
if index < len(self):
return self[index]
else:
return None
[docs]
def namedItem(self, name: str) -> Node | None:
"""Returns the specific node whose ID or, as a fallback, name matches the string specified by name."""
for item in self:
if item.id == name:
return item
elif item.name == name:
return item
return None
def __getitem__(self, index: int | str):
# can return dot notation i.e
# index = "named.item.with.periods" # TODO - test
if isinstance(index, str):
names = index.split(".")
if len(names) > 1:
return self.namedItem(names[0]).namedItem(".".join(names[1:]))
else:
return self.namedItem(index)
else:
return super().__getitem__(index)
MutationCallback = Callable[[list["MutationRecord"], "MutationObserver"], Any]
[docs]
class MutationRecord:
"""Single mutation payload delivered to a ``MutationObserver``.
Records describe one child-list, attribute, or character-data change and
carry the pieces of context the observer asked to receive.
"""
__slots__ = (
"type",
"target",
"addedNodes",
"removedNodes",
"previousSibling",
"nextSibling",
"attributeName",
"attributeNamespace",
"oldValue",
)
def __init__(
self,
type: str,
target: Node,
*,
addedNodes: Iterable[Node] = (),
removedNodes: Iterable[Node] = (),
previousSibling: Node | None = None,
nextSibling: Node | None = None,
attributeName: str | None = None,
attributeNamespace: str | None = None,
oldValue: str | None = None,
) -> None:
self.type = type
self.target = target
self.addedNodes = NodeList(addedNodes)
self.removedNodes = NodeList(removedNodes)
self.previousSibling = previousSibling
self.nextSibling = nextSibling
self.attributeName = attributeName
self.attributeNamespace = attributeNamespace
self.oldValue = oldValue
[docs]
class MutationObserver:
"""Observe DOM tree mutations and receive ``MutationRecord`` batches.
This implementation follows the familiar platform model: call ``observe()``
with a target and options, allow DOM operations to queue records, then
receive them through the callback or ``takeRecords()``.
"""
_all_observers: ClassVar[list["MutationObserver"]] = []
def __init__(self, callback: MutationCallback) -> None:
if not callable(callback):
raise TypeError("MutationObserver callback must be callable")
self.callback = callback
self._records: list[MutationRecord] = []
self._observations: dict[Node, dict[str, Any]] = {}
MutationObserver._all_observers.append(self)
def disconnect(self) -> None:
self._observations.clear()
self._records.clear()
def observe(self, target: Node, options: dict[str, Any]) -> None:
if not isinstance(target, Node):
raise TypeError("MutationObserver target must be a Node")
self._observations[target] = _normalize_mutation_observer_options(options)
def takeRecords(self) -> list[MutationRecord]:
records = list(self._records)
self._records.clear()
return records
def _enqueue_if_observing(self, record: MutationRecord) -> bool:
for current in _iter_ancestors_inclusive(record.target):
options = self._observations.get(current)
if options is None:
continue
if current is not record.target and not options["subtree"]:
continue
if record.type == "childList" and not options["childList"]:
continue
if record.type == "attributes":
if not options["attributes"]:
continue
attribute_filter = options.get("attributeFilter")
if attribute_filter is not None and record.attributeName not in attribute_filter:
continue
old_value = record.oldValue if options["attributeOldValue"] else None
filtered_record = MutationRecord(
"attributes",
record.target,
attributeName=record.attributeName,
attributeNamespace=record.attributeNamespace,
oldValue=old_value,
)
self._records.append(filtered_record)
return True
if record.type == "characterData":
if not options["characterData"]:
continue
old_value = record.oldValue if options["characterDataOldValue"] else None
filtered_record = MutationRecord("characterData", record.target, oldValue=old_value)
self._records.append(filtered_record)
return True
if record.type == "childList":
self._records.append(record)
return True
return False
def _flush(self) -> None:
if not self._records:
return
records = self.takeRecords()
self.callback(records, self)
[docs]
class ResizeObserverSize:
"""Inline and block dimensions reported by ``ResizeObserver`` entries."""
def __init__(self, inlineSize: float, blockSize: float) -> None:
self.inlineSize = inlineSize
self.blockSize = blockSize
[docs]
class ResizeObserverEntry:
"""Geometry snapshot for a single observed element resize."""
def __init__(self, target: Element, contentRect: DOMRectReadOnly) -> None:
self.target = target
self.contentRect = DOMRect.fromRect(contentRect)
size = ResizeObserverSize(self.contentRect.width, self.contentRect.height)
self.borderBoxSize = [size]
self.contentBoxSize = [size]
self.devicePixelContentBoxSize = [size]
ResizeObserverCallback = Callable[[list["ResizeObserverEntry"], "ResizeObserver"], Any]
[docs]
class ResizeObserver:
"""Observe element box changes through DOM geometry reads.
Domonic treats layout changes pragmatically: when relevant geometry changes
are computed, resize entries are queued and delivered to the callback.
"""
_all_observers: ClassVar[list["ResizeObserver"]] = []
def __init__(self, callback: ResizeObserverCallback) -> None:
if not callable(callback):
raise TypeError("ResizeObserver callback must be callable")
self.callback = callback
self._observations: dict[Element, tuple[float, float, float, float] | None] = {}
self._records: list[ResizeObserverEntry] = []
ResizeObserver._all_observers.append(self)
def observe(self, target: Element, options: dict[str, Any] | None = None) -> None:
if not isinstance(target, Element):
raise TypeError("ResizeObserver target must be an Element")
self._observations[target] = None
target.getBoundingClientRect()
def unobserve(self, target: Element) -> None:
self._observations.pop(target, None)
def disconnect(self) -> None:
self._observations.clear()
self._records.clear()
def takeRecords(self) -> list[ResizeObserverEntry]:
records = list(self._records)
self._records.clear()
return records
def _process(self, changed_target: Node | None = None, target_rect: DOMRectReadOnly | None = None) -> None:
for target, previous in list(self._observations.items()):
rect = DOMRect.fromRect(target_rect) if target is changed_target and target_rect is not None else DOMRect.fromRect(target.getBoundingClientRect())
current = (rect.x, rect.y, rect.width, rect.height)
if previous is None or previous != current:
self._observations[target] = current
self._records.append(ResizeObserverEntry(target, rect))
self._flush()
def _flush(self) -> None:
if not self._records:
return
records = self.takeRecords()
self.callback(records, self)
[docs]
class IntersectionObserverEntry:
"""Visibility snapshot for one target observed by ``IntersectionObserver``."""
def __init__(
self,
target: Element,
rootBounds: DOMRectReadOnly,
boundingClientRect: DOMRectReadOnly,
intersectionRect: DOMRectReadOnly,
time_value: float,
) -> None:
self.target = target
self.rootBounds = DOMRect.fromRect(rootBounds)
self.boundingClientRect = DOMRect.fromRect(boundingClientRect)
self.intersectionRect = DOMRect.fromRect(intersectionRect)
self.time = time_value
self.isIntersecting = self.intersectionRect.width > 0 and self.intersectionRect.height > 0
target_area = self.boundingClientRect.width * self.boundingClientRect.height
intersection_area = self.intersectionRect.width * self.intersectionRect.height
self.intersectionRatio = 0.0 if target_area == 0 else intersection_area / target_area
IntersectionObserverCallback = Callable[[list["IntersectionObserverEntry"], "IntersectionObserver"], Any]
[docs]
class IntersectionObserver:
"""Observe whether elements intersect a root rectangle or viewport-like area.
Domonic models intersections using element bounding boxes and a root
rectangle, which is enough for practical DOM-side visibility checks and
tests.
"""
_all_observers: ClassVar[list["IntersectionObserver"]] = []
def __init__(self, callback: IntersectionObserverCallback, options: dict[str, Any] | None = None) -> None:
if not callable(callback):
raise TypeError("IntersectionObserver callback must be callable")
self.callback = callback
self.root = (options or {}).get("root")
threshold = (options or {}).get("threshold", 0.0)
self.thresholds = sorted(threshold if isinstance(threshold, list) else [threshold])
self._observations: dict[Element, tuple[bool, float] | None] = {}
self._records: list[IntersectionObserverEntry] = []
IntersectionObserver._all_observers.append(self)
def observe(self, target: Element) -> None:
if not isinstance(target, Element):
raise TypeError("IntersectionObserver target must be an Element")
self._observations[target] = None
target.getBoundingClientRect()
def unobserve(self, target: Element) -> None:
self._observations.pop(target, None)
def disconnect(self) -> None:
self._observations.clear()
self._records.clear()
def takeRecords(self) -> list[IntersectionObserverEntry]:
records = list(self._records)
self._records.clear()
return records
def _process(self, changed_target: Node | None = None, target_rect: DOMRectReadOnly | None = None) -> None:
now_ms = time.perf_counter() * 1000.0
for target, previous in list(self._observations.items()):
bounding_rect = DOMRect.fromRect(target_rect) if target is changed_target and target_rect is not None else DOMRect.fromRect(target.getBoundingClientRect())
if isinstance(self.root, Element):
root_rect = DOMRect.fromRect(self.root.getBoundingClientRect())
else:
root_rect = DOMRect.fromRect(_default_intersection_root_rect(target, bounding_rect))
intersection_rect = _intersect_rects(root_rect, bounding_rect)
entry = IntersectionObserverEntry(target, root_rect, bounding_rect, intersection_rect, now_ms)
state = (entry.isIntersecting, entry.intersectionRatio)
if previous is None or previous != state:
self._observations[target] = state
self._records.append(entry)
self._flush()
def _flush(self) -> None:
if not self._records:
return
records = self.takeRecords()
self.callback(records, self)
class PerformanceEntry:
def __init__(self, name: str, entryType: str, startTime: float, duration: float) -> None:
self.name = name
self.entryType = entryType
self.startTime = startTime
self.duration = duration
def toJSON(self) -> dict[str, Any]:
return {
"name": self.name,
"entryType": self.entryType,
"startTime": self.startTime,
"duration": self.duration,
}
class PerformanceMark(PerformanceEntry):
def __init__(self, name: str, startTime: float) -> None:
super().__init__(name, "mark", startTime, 0.0)
class PerformanceMeasure(PerformanceEntry):
def __init__(self, name: str, startTime: float, duration: float) -> None:
super().__init__(name, "measure", startTime, duration)
PerformanceObserverCallback = Callable[[list["PerformanceEntry"], "PerformanceObserver"], Any]
class PerformanceObserver:
supportedEntryTypes: ClassVar[list[str]] = ["mark", "measure"]
_all_observers: ClassVar[list["PerformanceObserver"]] = []
def __init__(self, callback: PerformanceObserverCallback) -> None:
if not callable(callback):
raise TypeError("PerformanceObserver callback must be callable")
self.callback = callback
self._entry_types: set[str] = set()
self._records: list[PerformanceEntry] = []
PerformanceObserver._all_observers.append(self)
def observe(self, options: dict[str, Any]) -> None:
entry_types = options.get("entryTypes")
if not entry_types:
raise TypeError("PerformanceObserver.observe requires entryTypes")
self._entry_types = set(entry_types)
if options.get("buffered"):
try:
from domonic.javascript import performance as js_performance
for entry in js_performance.getEntries():
self._enqueue(entry)
except Exception:
pass
self._flush()
def disconnect(self) -> None:
self._entry_types.clear()
self._records.clear()
def takeRecords(self) -> list[PerformanceEntry]:
records = list(self._records)
self._records.clear()
return records
def _enqueue(self, entry: PerformanceEntry) -> None:
if entry.entryType in self._entry_types:
self._records.append(entry)
def _flush(self) -> None:
if not self._records:
return
records = self.takeRecords()
self.callback(records, self)
@classmethod
def _notify_entry(cls, entry: PerformanceEntry) -> None:
for observer in list(cls._all_observers):
observer._enqueue(entry)
observer._flush()
[docs]
class DOMException(Exception):
"""The DOMException interface represents an anormal event related to the DOM."""
INDEX_SIZE_ERR: int = 1
DOMSTRING_SIZE_ERR: int = 2
HIERARCHY_REQUEST_ERR: int = 3
WRONG_DOCUMENT_ERR: int = 4
INVALID_CHARACTER_ERR: int = 5
NO_DATA_ALLOWED_ERR: int = 6
NO_MODIFICATION_ALLOWED_ERR: int = 7
NOT_FOUND_ERR: int = 8
NOT_SUPPORTED_ERR: int = 9
INUSE_ATTRIBUTE_ERR: int = 10
INVALID_STATE_ERR: int = 11
SYNTAX_ERR: int = 12
INVALID_MODIFICATION_ERR: int = 13
NAMESPACE_ERR: int = 14
INVALID_ACCESS_ERR: int = 15
VALIDATION_ERR: int = 16
TYPE_MISMATCH_ERR: int = 17
SECURITY_ERR: int = 18
NETWORK_ERR: int = 19
ABORT_ERR: int = 20
URL_MISMATCH_ERR: int = 21
QUOTA_EXCEEDED_ERR: int = 22
TIMEOUT_ERR: int = 23
INVALID_NODE_TYPE_ERR: int = 24
DATA_CLONE_ERR: int = 25
def __init__(self, code, message: str | None = None) -> None:
self.code = code
self.message: str = message
self.name = "DOMException"
def __str__(self) -> str:
return self.message
def __repr__(self) -> str:
return self.message
[docs]
class DOMTimeStamp(int):
"""The DOMTimeStamp interface represents a numeric value which represents the
number of milliseconds since the epoch."""
def __init__(self, value):
self.value = value
def __str__(self):
return str(self.value)
def __repr__(self):
return str(self.value)
[docs]
class DOMPoint(vec3):
"""The DOMPoint interface represents a point specified by x and y coordinates."""
@staticmethod
def fromPoint(point: Any) -> "DOMPoint":
return DOMPoint(point.x, point.y, point.z, point.w)
def __init__(self, x: float, y: float, z: float = 0, w: float = 1) -> None:
self.x: float = x
self.y: float = y
self.z: float = z
self.w: float = w
super().__init__(x, y, z)
def __str__(self) -> str:
return "({}, {}, {}, {})".format(self.x, self.y, self.z, self.w)
def __repr__(self) -> str:
return "({}, {}, {}, {})".format(self.x, self.y, self.z, self.w)
[docs]
class DOMPointReadOnly(DOMPoint):
"""The DOMPointReadOnly interface represents a point specified by x and y coordinates."""
@staticmethod
def fromPoint(point: Any) -> "DOMPointReadOnly":
return DOMPointReadOnly(point.x, point.y, point.z, point.w)
def __init__(self, x: float, y: float, z: float = 0, w: float = 1) -> None:
self.x: float = x
self.y: float = y
self.z: float = z
self.w: float = w
super().__init__(x, y, z, w)
def __str__(self) -> str:
return "({}, {}, {}, {})".format(self.x, self.y, self.z, self.w)
def __repr__(self) -> str:
return "({}, {}, {}, {})".format(self.x, self.y, self.z, self.w)
[docs]
class DOMMatrixReadOnly:
"""Read-only 4x4 transformation matrix for DOM geometry APIs.
Supports the common 2D aliases as well as the full 4x4 member set used by
transforms, points, and animation/geometry helpers.
"""
@staticmethod
def fromFloat64Array(array: Iterable[float]) -> "DOMMatrixReadOnly":
return DOMMatrixReadOnly(*list(array))
@staticmethod
def fromFloat32Array(array: Iterable[float]) -> "DOMMatrixReadOnly":
return DOMMatrixReadOnly.fromFloat64Array(array)
@staticmethod
def fromMatrix(matrix: Any | None = None) -> "DOMMatrixReadOnly":
if matrix is None:
return DOMMatrixReadOnly()
values = []
for row in range(1, 5):
for col in range(1, 5):
values.append(getattr(matrix, f"m{row}{col}", 1.0 if row == col else 0.0))
return DOMMatrixReadOnly(*values)
def __init__(self, *values: float) -> None:
if not values:
values = (
1.0, 0.0, 0.0, 0.0,
0.0, 1.0, 0.0, 0.0,
0.0, 0.0, 1.0, 0.0,
0.0, 0.0, 0.0, 1.0,
)
if len(values) == 6:
a, b, c, d, e, f = values
values = (
a, b, 0.0, 0.0,
c, d, 0.0, 0.0,
0.0, 0.0, 1.0, 0.0,
e, f, 0.0, 1.0,
)
if len(values) != 16:
raise TypeError("DOMMatrix requires 6 or 16 values")
self._values = [float(value) for value in values]
def _get(self, row: int, col: int) -> float:
return self._values[(row - 1) * 4 + (col - 1)]
def _set(self, row: int, col: int, value: float) -> None:
self._values[(row - 1) * 4 + (col - 1)] = float(value)
@property
def is2D(self) -> bool:
return (
self.m13 == 0.0
and self.m14 == 0.0
and self.m23 == 0.0
and self.m24 == 0.0
and self.m31 == 0.0
and self.m32 == 0.0
and self.m34 == 0.0
and self.m43 == 0.0
and self.m33 == 1.0
and self.m44 == 1.0
)
@property
def isIdentity(self) -> bool:
return self._values == [
1.0, 0.0, 0.0, 0.0,
0.0, 1.0, 0.0, 0.0,
0.0, 0.0, 1.0, 0.0,
0.0, 0.0, 0.0, 1.0,
]
@property
def a(self) -> float:
return self.m11
@property
def b(self) -> float:
return self.m12
@property
def c(self) -> float:
return self.m21
@property
def d(self) -> float:
return self.m22
@property
def e(self) -> float:
return self.m41
@property
def f(self) -> float:
return self.m42
def __str__(self) -> str:
return f"DOMMatrix({', '.join(str(v) for v in self._values)})"
def toFloat64Array(self) -> list[float]:
return list(self._values)
def toFloat32Array(self) -> list[float]:
return list(self._values)
def toJSON(self) -> dict[str, float | bool]:
data = {f"m{row}{col}": self._get(row, col) for row in range(1, 5) for col in range(1, 5)}
data.update({"is2D": self.is2D, "isIdentity": self.isIdentity})
return data
def multiply(self, other: Any) -> "DOMMatrix":
return DOMMatrix.fromMatrix(self).multiplySelf(other)
def translate(self, tx: float = 0, ty: float = 0, tz: float = 0) -> "DOMMatrix":
return DOMMatrix.fromMatrix(self).translateSelf(tx, ty, tz)
def scale(self, scaleX: float = 1, scaleY: float | None = None, scaleZ: float = 1) -> "DOMMatrix":
return DOMMatrix.fromMatrix(self).scaleSelf(scaleX, scaleY, scaleZ)
def inverse(self) -> "DOMMatrix":
return DOMMatrix.fromMatrix(self).invertSelf()
def transformPoint(self, point: Any | None = None) -> DOMPoint:
if point is None:
point = DOMPoint(0, 0, 0, 1)
x = getattr(point, "x", 0.0)
y = getattr(point, "y", 0.0)
z = getattr(point, "z", 0.0)
w = getattr(point, "w", 1.0)
values = self._values
return DOMPoint(
x * values[0] + y * values[4] + z * values[8] + w * values[12],
x * values[1] + y * values[5] + z * values[9] + w * values[13],
x * values[2] + y * values[6] + z * values[10] + w * values[14],
x * values[3] + y * values[7] + z * values[11] + w * values[15],
)
for _row in range(1, 5):
for _col in range(1, 5):
setattr(
DOMMatrixReadOnly,
f"m{_row}{_col}",
property(lambda self, r=_row, c=_col: self._get(r, c)),
)
[docs]
class DOMMatrix(DOMMatrixReadOnly):
"""Mutable ``DOMMatrix`` implementation.
Use this when you want to construct, compose, invert, or transform points
with a matrix that can be updated in place.
"""
@staticmethod
def fromFloat64Array(array: Iterable[float]) -> "DOMMatrix":
return DOMMatrix(*list(array))
@staticmethod
def fromFloat32Array(array: Iterable[float]) -> "DOMMatrix":
return DOMMatrix.fromFloat64Array(array)
@staticmethod
def fromMatrix(matrix: Any | None = None) -> "DOMMatrix":
readonly = DOMMatrixReadOnly.fromMatrix(matrix)
return DOMMatrix(*readonly.toFloat64Array())
@DOMMatrixReadOnly.a.setter
def a(self, value: float) -> None:
self.m11 = value
@DOMMatrixReadOnly.b.setter
def b(self, value: float) -> None:
self.m12 = value
@DOMMatrixReadOnly.c.setter
def c(self, value: float) -> None:
self.m21 = value
@DOMMatrixReadOnly.d.setter
def d(self, value: float) -> None:
self.m22 = value
@DOMMatrixReadOnly.e.setter
def e(self, value: float) -> None:
self.m41 = value
@DOMMatrixReadOnly.f.setter
def f(self, value: float) -> None:
self.m42 = value
def multiplySelf(self, other: Any) -> "DOMMatrix":
other_matrix = DOMMatrixReadOnly.fromMatrix(other)
left = self._values
right = other_matrix._values
result = [0.0] * 16
for row in range(4):
for col in range(4):
result[row * 4 + col] = sum(left[row * 4 + k] * right[k * 4 + col] for k in range(4))
self._values = result
return self
def translateSelf(self, tx: float = 0, ty: float = 0, tz: float = 0) -> "DOMMatrix":
translation = DOMMatrix(
1, 0, 0, 0,
0, 1, 0, 0,
0, 0, 1, 0,
tx, ty, tz, 1,
)
return self.multiplySelf(translation)
def scaleSelf(self, scaleX: float = 1, scaleY: float | None = None, scaleZ: float = 1) -> "DOMMatrix":
if scaleY is None:
scaleY = scaleX
scale = DOMMatrix(
scaleX, 0, 0, 0,
0, scaleY, 0, 0,
0, 0, scaleZ, 0,
0, 0, 0, 1,
)
return self.multiplySelf(scale)
def invertSelf(self) -> "DOMMatrix":
matrix = [[self._values[row * 4 + col] for col in range(4)] for row in range(4)]
identity = [[1.0 if row == col else 0.0 for col in range(4)] for row in range(4)]
for col in range(4):
pivot = max(range(col, 4), key=lambda row: abs(matrix[row][col]))
if matrix[pivot][col] == 0:
raise ValueError("Matrix is not invertible")
matrix[col], matrix[pivot] = matrix[pivot], matrix[col]
identity[col], identity[pivot] = identity[pivot], identity[col]
factor = matrix[col][col]
matrix[col] = [value / factor for value in matrix[col]]
identity[col] = [value / factor for value in identity[col]]
for row in range(4):
if row == col:
continue
factor = matrix[row][col]
matrix[row] = [current - factor * pivot_value for current, pivot_value in zip(matrix[row], matrix[col])]
identity[row] = [current - factor * pivot_value for current, pivot_value in zip(identity[row], identity[col])]
self._values = [identity[row][col] for row in range(4) for col in range(4)]
return self
for _row in range(1, 5):
for _col in range(1, 5):
readonly_prop = getattr(DOMMatrixReadOnly, f"m{_row}{_col}")
setattr(
DOMMatrix,
f"m{_row}{_col}",
readonly_prop.setter(lambda self, value, r=_row, c=_col: self._set(r, c, value)),
)
[docs]
class DOMQuad:
"""The DOMQuad interface represents a quadrilateral on the plane with its
four corners represented as Cartesian coordinates."""
@staticmethod
def fromRect(rect: DOMRect) -> "DOMQuad":
return DOMQuad(
DOMPointReadOnly(rect.left, rect.top),
DOMPointReadOnly(rect.right, rect.top),
DOMPointReadOnly(rect.right, rect.bottom),
DOMPointReadOnly(rect.left, rect.bottom),
)
@staticmethod
def fromQuad(quad: Any) -> "DOMQuad":
return DOMQuad(quad.p1.x, quad.p1.y, quad.p2.x, quad.p2.y, quad.p3.x, quad.p3.y, quad.p4.x, quad.p4.y)
@staticmethod
def getBounds(quad: "DOMQuad") -> DOMRect:
xs = [quad.p1.x, quad.p2.x, quad.p3.x, quad.p4.x]
ys = [quad.p1.y, quad.p2.y, quad.p3.y, quad.p4.y]
left = min(xs)
top = min(ys)
right = max(xs)
bottom = max(ys)
return DOMRect(left, top, right - left, bottom - top)
@staticmethod
def toJSON(quad: DOMQuad) -> dict[str, dict[str, float]]:
return {
"p1": {"x": quad.p1.x, "y": quad.p1.y},
"p2": {"x": quad.p2.x, "y": quad.p2.y},
"p3": {"x": quad.p3.x, "y": quad.p3.y},
"p4": {"x": quad.p4.x, "y": quad.p4.y},
}
def __init__(self, p1: Any, p2: Any, p3: Any, p4: Any) -> None:
self.p1 = p1
self.p2 = p2
self.p3 = p3
self.p4 = p4
def __str__(self):
return "({}, {}, {}, {})".format(self.p1, self.p2, self.p3, self.p4)
# NodeFilter
# from xml.dom.NodeFilter import NodeFilter
# https://bspaans.github.io/python-mingus/_modules/xml/dom/xmlbuilder.html
# https://www.w3.org/TR/2003/WD-DOM-Level-3-LS-20030226/load-save.html
# https://bspaans.github.io/python-mingus/_modules/xml/dom/xmlbuilder.html
class NodeFilter:
SHOW_ALL: int = 0xFFFFFFFF
SHOW_ELEMENT: int = 0x00000001
SHOW_ATTRIBUTE: int = 0x00000002
SHOW_TEXT: int = 0x00000004
SHOW_CDATA_SECTION: int = 0x00000008
SHOW_ENTITY_REFERENCE: int = 0x00000010
SHOW_ENTITY: int = 0x00000020
SHOW_PROCESSING_INSTRUCTION: int = 0x00000040
SHOW_COMMENT: int = 0x00000080
SHOW_DOCUMENT: int = 0x00000100
SHOW_DOCUMENT_TYPE: int = 0x00000200
SHOW_DOCUMENT_FRAGMENT: int = 0x00000400
SHOW_NOTATION: int = 0x00000800
FILTER_ACCEPT: int = 1
FILTER_REJECT: int = 2
FILTER_SKIP: int = 3
# def acceptNode(node):
# return NodeFilter.FILTER_ACCEPT
# return node
# def acceptNode(node):
# result
# if active:
# raise Exception('DOMException: INVALID_STATE_ERR')
# active = True
# result = filter(node)
# active = False
# return result
[docs]
class NodeIterator:
"""[NodeIterator is an iterator object that iterates over the descendants of a node, in tree order.]"""
def __init__(
self,
root: Node,
whatToShow: int = NodeFilter.SHOW_ALL,
filter: Any = None,
entityReferenceExpansion: bool = False,
) -> None:
self.root = root
self.whatToShow = whatToShow
self._filter = filter
self.entityReferenceExpansion = entityReferenceExpansion
self.node = root
self.pointer = -1
self.stack: list[Node] = []
def collect(node: Node) -> None:
self.stack.append(node)
for child in getattr(node, "childNodes", []):
if isinstance(child, str):
continue
collect(child)
collect(root)
@property
def filter(self) -> Any:
return self._filter
# def expandEntityReferences(self, expand):
# Is a boolean value indicating if,
# when discarding an EntityReference its whole sub-tree must be discarded at the same time.
[docs]
def referenceNode(self) -> Node:
"""Returns the Node that is being iterated over."""
return self.node
[docs]
def pointerBeforeReferenceNode(self) -> bool:
"""Returns a boolean flag that indicates whether the NodeIterator
is anchored before, the flag being true,
or after, the flag being false, the anchor node.
"""
return self.pointer < 0
def detach(self) -> None:
# This operation is a no-op. It doesn't do anything.
# Previously it was telling the engine that the NodeIterator was no more used, but this is now useless.
pass
[docs]
def previousNode(self) -> Node | None:
"""Returns the previous Node in the document, or null if there are none."""
if self.pointer <= 0:
return None
self.pointer -= 1
self.node = self.stack[self.pointer]
return self.node
[docs]
def nextNode(self) -> Node | None:
"""Returns the next Node in the document, or null if there are none."""
self.pointer += 1
while self.pointer < len(self.stack):
candidate = self.stack[self.pointer]
if nodeFilter(self, candidate) == NodeFilter.FILTER_ACCEPT:
self.node = candidate
return candidate
self.pointer += 1
return None
mapChild = {"first": "firstChild", "last": "lastChild", "next": "firstChild", "previous": "lastChild"}
mapSibling = {"next": "nextSibling", "previous": "previousSibling"}
# toString = mapChild.toString
# def _is(x, _type):
# print('!!!!!!!!!!!!!!!!!!! comparing', x, _type)
# return mapChild[x].toLowerCase() == '[object ' + _type.toLowerCase() + ']'
def nodeFilter(tw: NodeIterator | TreeWalker, node: Node) -> int:
# Maps nodeType to whatToShow
# print(node, type(node))
# if isinstance(node, (str)): #, Text)):
# node = Text(node)
# return NodeFilter.FILTER_SKIP
# return NodeFilter.FILTER_REJECT
if not (((1 << (node.nodeType - 1)) & tw.whatToShow)):
return NodeFilter.FILTER_SKIP
if tw._filter == None:
return NodeFilter.FILTER_ACCEPT
if callable(tw._filter):
return tw._filter(node)
return tw._filter.acceptNode(node)
def str_to_TextNode(content_str: Any) -> Any:
if isinstance(content_str, str):
return Text(content_str)
return content_str
def traverseChildren(tw: TreeWalker, _type: str) -> Node | None:
# var child, node, parent, result, sibling
# print('mapChild[_type]', mapChild[_type])
node = getattr(
tw.currentNode, mapChild[_type]
) # TODO - allow dict access to node props?.... tw.currentNode[mapChild[_type]]
# print('tw.currentNode', tw.currentNode)
# node = str_to_TextNode(node)
while node != None:
# node = str_to_TextNode(node)
result = nodeFilter(tw, node)
if result == NodeFilter.FILTER_ACCEPT:
tw.currentNode = node
return node
if result == NodeFilter.FILTER_SKIP:
child = getattr(node, mapChild[_type])
if child != None:
node = child
continue
while node != None:
sibling = getattr(node, mapSibling["next" if _type == "first" else "previous"])
if sibling != None:
node = sibling
break
parent = node.parentNode
if parent == None or parent == tw.root or parent == tw.currentNode:
return None
else:
node = parent
return None
def traverseSiblings(tw: TreeWalker, type: str) -> Node | None:
# node, result, sibling
node = tw.currentNode
if node == tw.root:
return None
while True:
sibling = getattr(node, mapSibling[type])
while sibling != None:
node = sibling
result = nodeFilter(tw, node)
if result == NodeFilter.FILTER_ACCEPT:
tw.currentNode = node
return node
sibling = getattr(node, mapChild[type])
if result == NodeFilter.FILTER_REJECT:
sibling = getattr(node, mapSibling[type])
node = node.parentNode
if node == None or node == tw.root:
return None
if nodeFilter(tw, node) == NodeFilter.FILTER_ACCEPT:
return None
def nextSkippingChildren(node: Node, stayWithin: Node) -> Node | None:
# if isinstance(node, str):
# node = Text(node)
# return None
# TODO - casting is not enough. as the Text node does not know its siblings
# print( "nsc", node, "??", stayWithin )
# print( "AND:", node.nextSibling )
if node == stayWithin:
# print('a')
return None
if node.nextSibling != None:
# print('b')
return node.nextSibling
while node.parentNode != None:
# print('c')
node = node.parentNode
if node == stayWithin:
# print('d')
return None
if node.nextSibling != None:
# print('e')
return node.nextSibling
return None
# https://developer.mozilla.org/en-US/docs/Web/API/TreeWalker
[docs]
class TreeWalker:
"""The TreeWalker object represents the nodes of a document subtree and a position within them."""
def _upgrade_dom(self) -> None:
"""[
Our dom has some strings that are not Text Nodes
so we have to upgrade them to Node objects. As we can't know siblings otherwise
# TODO - consider upgrading as they are created.
]
"""
def upgrade(el: Node) -> None:
if isinstance(el, (Text, str)):
return
for child in el:
if isinstance(child, str):
# print('doin one')
newchild = Text(child)
el.replaceChild(newchild, child)
newchild.parentNode = el
self._root._iterate(self._root, upgrade)
def __init__(
self,
node: Node,
whatToShow: int = NodeFilter.SHOW_ALL,
_filter: Any = None,
expandEntityReferences: bool = False,
) -> None:
self._root = node
self._upgrade_dom()
# print("test", type(self._root[0][0]))
self.currentNode = node
# TODO - convert whatToShow to a number?
self.whatToShow = whatToShow
# self.whatToShow = self.whatToShow & 0xFFFFFFFF
self.whatToShow = whatToShow or 0
self._filter = _filter
def acceptNode(node: Node) -> int:
nonlocal _filter
# result
# if active:
# raise Exception('DOMException: INVALID_STATE_ERR')
# active = True
result = _filter(node)
# active = False
return result
if self._filter is not None:
NodeFilter.acceptNode = acceptNode
self.last = None
self.parent = None
self.previous = None
self.children: list[Node] = []
self.childIndex = 0
self.tree = None
""" Is a boolean value indicating,
when discarding an entity reference its whole sub-tree must be discarded at the same time. """
self.expandEntityReferences = expandEntityReferences
@property
def root(self) -> Node:
"""Returns a Node representing the root node as specified when the TreeWalker was created."""
return self._root
[docs]
def whatToShow(self, options: int) -> int:
"""Returns an unsigned long being a bitmask made of constants describing the types of Node that must be presented.
Non-matching nodes are skipped, but their children may be included, if relevant. The possible values are:"""
return options
# def filter(self, options):
# """ Returns a NodeFilter object that can be used to filter the nodes that the TreeWalker visits. """
# return options
# @property
# def currentNode(self):
# """ Is the Node on which the TreeWalker is currently pointing at. """
# return self.currentNode
[docs]
def parentNode(self) -> Node | None:
"""Moves the current Node to the first visible ancestor node in the document order,
and returns the found node. It also moves the current node to this one. If no such node exists,
or if it is before that the root node defined at the object construction,
returns null and the current node is not changed."""
# return self.currentNode.parentNode
node = self.currentNode
while node != None and node != self.root:
node = node.parentNode
if node != None and nodeFilter(self, node) == NodeFilter.FILTER_ACCEPT:
self.currentNode = node
return node
return None
[docs]
def firstChild(self) -> Node | None:
"""Moves the current Node to the first visible child of the current node, and returns the found child.
It also moves the current node to this child. If no such child exists,
returns null and the current node is not changed."""
# return self.currentNode.firstChild
return traverseChildren(self, "first")
[docs]
def lastChild(self) -> Node | None:
"""Moves the current Node to the last visible child of the current node, and returns the found child.
It also moves the current node to this child.
If no such child exists, null is returned and the current node is not changed."""
# return self.currentNode.lastChild
return traverseChildren(self, "last")
[docs]
def previousSibling(self) -> Node | None:
"""Moves the current Node to its previous sibling, if any, and returns the found sibling.
If there is no such node, return null and the current node is not changed.
"""
# return self.previous
return traverseSiblings(self, "previous")
[docs]
def nextSibling(self) -> Node | None:
"""Moves the current Node to its next sibling, if any, and returns the found sibling.
If there is no such node, null is returned and the current node is not changed."""
# return self.currentNode.nextSibling
return traverseSiblings(self, "next")
[docs]
def previousNode(self):
"""Moves the current Node to the previous visible node in the document order,
and returns the found node. It also moves the current node to this one.
If no such node exists, or if it is before that the root node defined at the object construction,
returns null and the current node is not changed."""
# return self.previous
# raise NotImplementedError()
# var node, result, sibling
node = self.currentNode
while node != self.root:
sibling = node.previousSibling
while sibling != None:
node = sibling
result = nodeFilter(self, node)
while result != NodeFilter.FILTER_REJECT and node.lastChild != None:
node = node.lastChild
result = nodeFilter(self, node)
if result == NodeFilter.FILTER_ACCEPT:
self.currentNode = node
return node
if node == self.root or node.parentNode == None:
return None
node = node.parentNode
if nodeFilter(self, node) == NodeFilter.FILTER_ACCEPT:
self.currentNode = node
return node
return None
[docs]
def nextNode(self):
"""Moves the current Node to the next visible node in the document order, and returns the found node.
It also moves the current node to this one.
If no such node exists, returns None and the current node is not changed.
can be used in a while loop to iterate over all the nodes in the document order.
"""
# var node, result, following;
node = self.currentNode
if isinstance(node, str):
node = Text(node)
# return node
result = NodeFilter.FILTER_ACCEPT
while True:
# print('rrr:::', result, node)
if isinstance(node, str):
Text(node)
# continue
while result != NodeFilter.FILTER_REJECT and node.firstChild != None:
# print('rrr222:::', result, node)
node = node.firstChild
if isinstance(node, str):
node = Text(node)
# result = NodeFilter.FILTER_REJECT
# continue
# break
# return None
result = nodeFilter(self, node)
if result == NodeFilter.FILTER_ACCEPT:
self.currentNode = node
return node
following = nextSkippingChildren(node, self.root)
if following != None:
node = following
else:
# print('NONE')
return None
result = nodeFilter(self, node)
if result == NodeFilter.FILTER_ACCEPT:
self.currentNode = node
return node
# TODO - create fetch package and move the js fetch stuff to it?
# fetch api
# AbortController
# AbortSignal
# Cache
# CacheStorage
# ContentIndex
# ContactPicker
# Client - serviceworker api
# CredentialsContainer - new login api
# DOMMatrix #https://developer.mozilla.org/en-US/docs/Web/API/DOMMatrix
# DOMParser
# IndexedDB API
# ImageBitmap
# ImageBitmapRenderingContext
# ImageData
# MutationObserver
# MutationRecord
# OverconstrainedError
# QueueingStrategy
# ReadableStream
# SCTP
# SourceBuffer
# SourceBufferAppendMode
# SourceBufferAppendWindowEnd
# TimeRanges - media
# TrackEvent - media
# ValidityState
# Web Share API
# WebGL
# also
# https://developer.mozilla.org/en-US/docs/Mozilla/Add-ons/WebExtensions/API
# HTMLElementTagNameMap < how many of these types are there?
# XMLSerializer = xml.dom.minidom.XMLSerializer?
# XMLSerializer.serializeToString(rootNode)
class Sanitizer:
def __init__(self, rules=None, *args, **kwargs):
"""Creates and returns a Sanitizer object."""
# casting as object gives us . notation
from domonic.javascript import Object
self._default_configuration = Object(
{
"allowCustomElements": False,
# "allowElements": [], # elements that the sanitizer should retain in the input.
"blockElements": [], # elements where the sanitizer should remove the elements from the input, but retain their children.
"dropElements": [], # elements that the sanitizer should remove from the input, including its children.
# "allowAttributes": [], # determines whether an attribute (on a given element) should be allowed.
"dropAttributes": [], # determines whether an attribute (on a given element) should be dropped.
"allowCustomElements": False, # determines whether custom elements are to be considered. The default is to drop them. If this option is true, custom elements will still be checked against all other built-in or configured configured checks.
"allowComments": False, # determines whether HTML comments are allowed.
"allowElements": [
"a",
"abbr",
"acronym",
"address",
"area",
"article",
"aside",
"audio",
"b",
"bdi",
"bdo",
"bgsound",
"big",
"blockquote",
"body",
"br",
"button",
"canvas",
"caption",
"center",
"cite",
"code",
"col",
"colgroup",
"datalist",
"dd",
"del",
"details",
"dfn",
"dialog",
"dir",
"div",
"dl",
"dt",
"em",
"fieldset",
"figcaption",
"figure",
"font",
"footer",
"form",
"h1",
"h2",
"h3",
"h4",
"h5",
"h6",
"head",
"header",
"hgroup",
"hr",
"html",
"i",
"img",
"input",
"ins",
"kbd",
"keygen",
"label",
"layer",
"legend",
"li",
"link",
"listing",
"main",
"map",
"mark",
"marquee",
"menu",
"meta",
"meter",
"nav",
"nobr",
"ol",
"optgroup",
"option",
"output",
"p",
"picture",
"popup",
"pre",
"progress",
"q",
"rb",
"rp",
"rt",
"rtc",
"ruby",
"s",
"samp",
"section",
"select",
"selectmenu",
"small",
"source",
"span",
"strike",
"strong",
"style",
"sub",
"summary",
"sup",
"table",
"tbody",
"td",
"tfoot",
"th",
"thead",
"time",
"tr",
"track",
"tt",
"u",
"ul",
"var",
"video",
"wbr",
],
"allowAttributes": {
"abbr": ["*"],
"accept": ["*"],
"accept-charset": ["*"],
"accesskey": ["*"],
"action": ["*"],
"align": ["*"],
"alink": ["*"],
"allow": ["*"],
"allowfullscreen": ["*"],
"alt": ["*"],
"anchor": ["*"],
"archive": ["*"],
"as": ["*"],
"async": ["*"],
"autocapitalize": ["*"],
"autocomplete": ["*"],
"autocorrect": ["*"],
"autofocus": ["*"],
"autopictureinpicture": ["*"],
"autoplay": ["*"],
"axis": ["*"],
"background": ["*"],
"behavior": ["*"],
"bgcolor": ["*"],
"border": ["*"],
"bordercolor": ["*"],
"capture": ["*"],
"cellpadding": ["*"],
"cellspacing": ["*"],
"challenge": ["*"],
"char": ["*"],
"charoff": ["*"],
"charset": ["*"],
"checked": ["*"],
"cite": ["*"],
"class": ["*"],
"classid": ["*"],
"clear": ["*"],
"code": ["*"],
"codebase": ["*"],
"codetype": ["*"],
"color": ["*"],
"cols": ["*"],
"colspan": ["*"],
"compact": ["*"],
"content": ["*"],
"contenteditable": ["*"],
"controls": ["*"],
"controlslist": ["*"],
"conversiondestination": ["*"],
"coords": ["*"],
"crossorigin": ["*"],
"csp": ["*"],
"data": ["*"],
"datetime": ["*"],
"declare": ["*"],
"decoding": ["*"],
"default": ["*"],
"defer": ["*"],
"dir": ["*"],
"direction": ["*"],
"dirname": ["*"],
"disabled": ["*"],
"disablepictureinpicture": ["*"],
"disableremoteplayback": ["*"],
"disallowdocumentaccess": ["*"],
"download": ["*"],
"draggable": ["*"],
"elementtiming": ["*"],
"enctype": ["*"],
"end": ["*"],
"enterkeyhint": ["*"],
"event": ["*"],
"exportparts": ["*"],
"face": ["*"],
"for": ["*"],
"form": ["*"],
"formaction": ["*"],
"formenctype": ["*"],
"formmethod": ["*"],
"formnovalidate": ["*"],
"formtarget": ["*"],
"frame": ["*"],
"frameborder": ["*"],
"headers": ["*"],
"height": ["*"],
"hidden": ["*"],
"high": ["*"],
"href": ["*"],
"hreflang": ["*"],
"hreftranslate": ["*"],
"hspace": ["*"],
"http-equiv": ["*"],
"id": ["*"],
"imagesizes": ["*"],
"imagesrcset": ["*"],
"importance": ["*"],
"impressiondata": ["*"],
"impressionexpiry": ["*"],
"incremental": ["*"],
"inert": ["*"],
"inputmode": ["*"],
"integrity": ["*"],
"invisible": ["*"],
"is": ["*"],
"ismap": ["*"],
"keytype": ["*"],
"kind": ["*"],
"label": ["*"],
"lang": ["*"],
"language": ["*"],
"latencyhint": ["*"],
"leftmargin": ["*"],
"link": ["*"],
"list": ["*"],
"loading": ["*"],
"longdesc": ["*"],
"loop": ["*"],
"low": ["*"],
"lowsrc": ["*"],
"manifest": ["*"],
"marginheight": ["*"],
"marginwidth": ["*"],
"max": ["*"],
"maxlength": ["*"],
"mayscript": ["*"],
"media": ["*"],
"method": ["*"],
"min": ["*"],
"minlength": ["*"],
"multiple": ["*"],
"muted": ["*"],
"name": ["*"],
"nohref": ["*"],
"nomodule": ["*"],
"nonce": ["*"],
"noresize": ["*"],
"noshade": ["*"],
"novalidate": ["*"],
"nowrap": ["*"],
"object": ["*"],
"open": ["*"],
"optimum": ["*"],
"part": ["*"],
"pattern": ["*"],
"ping": ["*"],
"placeholder": ["*"],
"playsinline": ["*"],
"policy": ["*"],
"poster": ["*"],
"preload": ["*"],
"pseudo": ["*"],
"readonly": ["*"],
"referrerpolicy": ["*"],
"rel": ["*"],
"reportingorigin": ["*"],
"required": ["*"],
"resources": ["*"],
"rev": ["*"],
"reversed": ["*"],
"role": ["*"],
"rows": ["*"],
"rowspan": ["*"],
"rules": ["*"],
"sandbox": ["*"],
"scheme": ["*"],
"scope": ["*"],
"scopes": ["*"],
"scrollamount": ["*"],
"scrolldelay": ["*"],
"scrolling": ["*"],
"select": ["*"],
"selected": ["*"],
"shadowroot": ["*"],
"shadowrootdelegatesfocus": ["*"],
"shape": ["*"],
"size": ["*"],
"sizes": ["*"],
"slot": ["*"],
"span": ["*"],
"spellcheck": ["*"],
"src": ["*"],
"srcdoc": ["*"],
"srclang": ["*"],
"srcset": ["*"],
"standby": ["*"],
"start": ["*"],
"step": ["*"],
"style": ["*"],
"summary": ["*"],
"tabindex": ["*"],
"target": ["*"],
"text": ["*"],
"title": ["*"],
"topmargin": ["*"],
"translate": ["*"],
"truespeed": ["*"],
"trusttoken": ["*"],
"type": ["*"],
"usemap": ["*"],
"valign": ["*"],
"value": ["*"],
"valuetype": ["*"],
"version": ["*"],
"virtualkeyboardpolicy": ["*"],
"vlink": ["*"],
"vspace": ["*"],
"webkitdirectory": ["*"],
"width": ["*"],
"wrap": ["*"],
},
}
)
self.config = None
if isinstance(rules, dict):
self.rules = rules
# create a new configuration which is a copy of the default but change it based on the rules object
import copy
self.config = copy.deepcopy(self._default_configuration)
for key, value in self.rules.items():
# print('ADDING RULES', key, value)
self.config[key] = value
else:
self.rules = None
self.config = self._default_configuration
def getDefaultConfiguration(self):
return self._default_configuration
def getConfiguration(self):
"""[return the configuration object]
Returns:
[Object]: [an Object with the users configuration]
"""
return self.config
def sanitize(self, frag):
"""Returns a sanitized DocumentFragment from an input, removing any offending elements or attributes."""
if isinstance(frag, str):
# parse to html then remove all the bad stuff?? - is a really bad idea. as it goes through eval.
from domonic import domonic
frag = domonic.load(frag)
isDomNode = False
if isinstance(frag, Document):
isDomNode = True
if not isDomNode:
newfrag = Document.createDocumentFragment()
if isinstance(frag, (tuple, list)):
for f in frag:
newfrag.appendChild(f)
else:
newfrag.appendChild(frag)
frag = newfrag
# TODO "allowCustomElements": # "allowElements": [], # "blockElements": [], # "dropElements": [], # "allowAttributes": [],
# TODO "dropAttributes": # "allowCustomElements": # "allowComments": # "allowElements" # allowAttributes
for t in self.config["dropElements"]:
el = frag.getElementsByTagName(t)
el.parentNode.removeChild(el)
for t in self.config["dropAttributes"]:
for e in self.config["allowElements"]:
els = frag.getElementsByTagName(e)
if els != False and len(els) > 0:
for el in els:
for each in el.attributes:
if each.name == t:
el.removeAttribute(each.name)
# print("test" frag.querySelectorAll('span'))
# print("test2", frag.getElementsByTagName('span'))
for e in self.config["allowElements"]:
els = frag.getElementsByTagName(e)
if els != False and len(els) > 0:
for el in els:
# print(el, el.kwargs, el.attributes, el.__attributes__, type(el.attributes))
for each in el.attributes:
# print(each)
key = each.name
val = each.value
# print(key, val)
allowed_on = self.config["allowAttributes"].get(key)
# print("ALLOWED ON:", key, allowed_on)
if allowed_on == None:
el.removeAttribute(key)
continue
if "*" in allowed_on:
continue
if e not in allowed_on:
el.removeAttribute(key)
# else:
# print(key + ' is allowed')
for t in self.config["blockElements"]:
el = frag.getElementsByTagName(str(t))
# keep the children of the element and add them back to the parent
for c in el.childNodes:
frag.parentNode.appendChild(c)
# remove the element
frag.parentNode.removeChild(el)
# print(type(frag))
return frag
def sanitizeToString(self, frag) -> str:
"""Returns a sanitized String from an input, removing any offending elements or attributes."""
return str(self.sanitize(frag))
[docs]
class HTMLElement(Element):
name = ""
[docs]
class HTMLAnchorElement(HTMLElement):
name = "a"
def __init__(self, *args, href=None, target=None, rel=None, download=None, type=None, **kwargs):
"""HTMLAnchorElement
Args:
href (str, optional): Specifies the URL of the page the link goes to.
target (str, optional): Specifies where to open the linked document (e.g., "_self", "_blank").
rel (str, optional): Specifies the relationship between the current document and the linked document (e.g., "noopener", "noreferrer").
download (str, optional): Specifies that the target will be downloaded when clicked, instead of navigating to it.
type (str, optional): Specifies the MIME type of the linked resource.
"""
super().__init__(*args, **kwargs)
if href is not None:
self.setAttribute("href", href)
if target is not None:
self.setAttribute("target", target)
if rel is not None:
self.setAttribute("rel", rel)
if download is not None:
self.setAttribute("download", download)
if type is not None:
self.setAttribute("type", type)
[docs]
class HTMLAreaElement(HTMLElement):
name = "area"
def __init__(self, *args, href=None, target=None, alt=None, coords=None, shape=None, **kwargs):
"""HTMLAreaElement
Args:
href (str, optional): Specifies the URL of the page the area links to.
target (str, optional): Specifies where to open the linked document (e.g., "_self", "_blank").
alt (str, optional): Specifies alternative text for the area.
coords (str, optional): Specifies the coordinates of the area in the image map (e.g., "x1,y1,x2,y2").
shape (str, optional): Specifies the shape of the clickable area (e.g., "rect", "circle", "poly").
"""
super().__init__(*args, **kwargs)
if href is not None:
self.setAttribute("href", href)
if target is not None:
self.setAttribute("target", target)
if alt is not None:
self.setAttribute("alt", alt)
if coords is not None:
self.setAttribute("coords", coords)
if shape is not None:
self.setAttribute("shape", shape)
[docs]
class HTMLAudioElement(HTMLElement):
name = "audio"
def __init__(
self, *args, autoplay: bool = None, controls=None, loop=None, muted=None, preload=None, src=None, **kwargs
):
"""HTMLAudioElement
Args:
autoplay (bool, optional): if specified, the audio will automatically begin playback as soon as it can do so, without waiting for the entire audio file to finish downloading
controls (_type_, optional): _description_. Defaults to None.
loop (_type_, optional): _description_. Defaults to None.
muted (_type_, optional): _description_. Defaults to None.
preload (_type_, optional): _description_. Defaults to None.
src (_type_, optional): _description_. Defaults to None.
"""
super().__init__(*args, **kwargs)
if autoplay is not None:
self.setAttribute("autoplay", autoplay)
if controls is not None:
self.setAttribute("controls", controls)
if loop is not None:
self.setAttribute("loop", loop)
if muted is not None:
self.setAttribute("muted", muted)
if preload is not None:
self.setAttribute("preload", preload)
if src is not None:
self.setAttribute("src", src)
[docs]
class HTMLBRElement(HTMLElement):
name = "br"
__isempty = True
def __str__(self):
if DOMConfig.RENDER_OPTIONAL_CLOSING_SLASH:
if DOMConfig.SPACE_BEFORE_OPTIONAL_CLOSING_SLASH:
return f"<{self.name}{self.__attributes__} />"
else:
return f"<{self.name}{self.__attributes__}/>"
return f"<{self.name}{self.__attributes__} >"
[docs]
class HTMLBaseElement(HTMLElement):
name = "base"
def __init__(self, *args, href=None, target=None, **kwargs):
"""HTMLBaseElement
Args:
href (str, optional): The base URL to be used throughout the document for relative URLs. Absolute and relative URLs are allowed.
target (str, optional): A keyword or author-defined name of the default browsing context...
"""
super().__init__(*args, **kwargs)
if href is not None:
self.setAttribute("href", href)
if target is not None:
self.setAttribute("target", target)
[docs]
class HTMLBaseFontElement(HTMLElement):
name = "basefont"
def __init__(self, *args, color=None, face=None, size=None, **kwargs):
super().__init__(*args, **kwargs)
if color is not None:
self.setAttribute("color", color)
if face is not None:
self.setAttribute("face", face)
if size is not None:
self.setAttribute("size", size)
[docs]
class HTMLBodyElement(HTMLElement):
name = "body"
def __init__(
self,
*args,
aLink=None,
background=None,
bgColor=None,
link=None,
onload=None,
onunload=None,
text=None,
vLink=None,
**kwargs,
):
"""HTMLBodyElement
Appears docs are telling you not to use many of the props you can pass and to use css instead.
Args:
aLink (str, optional): Color of text for hyperlinks when selected. Do not use this attribute! Use the CSS color property in conjunction with the :active pseudo-class instead.
background (str, optional): URI of a image to use as a background. Do not use this attribute! Use the CSS background property on the element instead.
bgColor (str, optional): Background color for the document. Do not use this attribute! Use the CSS background-color property on the element instead.
bgProperties (str, optional): The size of the text.
link (str, optional): Color of text for unvisited hypertext links. Do not use this attribute! Use the CSS color property in conjunction with the :link pseudo-class instead.
onload (str, optional): Function to call when the document is going away.
onunload (str, optional): Function to call when the document has finished loading.
text (str, optional): Foreground color of text. Do not use this attribute! Use CSS color property on the element instead.
vLink (str, optional): Color of text for visited hypertext links. Do not use this attribute! Use the CSS color property in conjunction with the :visited pseudo-class instead.
"""
super().__init__(*args, **kwargs)
if aLink is not None:
self.setAttribute("aLink", aLink)
if background is not None:
self.setAttribute("background", background)
if bgColor is not None:
self.setAttribute("bgColor", bgColor)
if link is not None:
self.setAttribute("link", link)
if onload is not None:
self.setAttribute("onload", onload)
if onunload is not None:
self.setAttribute("onunload", onunload)
if text is not None:
self.setAttribute("text", text)
if vLink is not None:
self.setAttribute("vLink", vLink)
[docs]
class HTMLCanvasElement(HTMLElement):
name = "canvas"
def __init__(self, *args, width: int = None, height: int = None, **kwargs):
"""HTMLCanvasElement
Args:
width (int, optional): The height of the coordinate space in CSS pixels. Defaults to 150.
height (int, optional): The width of the coordinate space in CSS pixels. Defaults to 300.
"""
super().__init__(*args, **kwargs)
if width is not None:
self.setAttribute("width", width)
if height is not None:
self.setAttribute("height", height)
[docs]
class HTMLContentElement(HTMLElement):
name = "content"
def __init__(self, *args, select=None, **kwargs):
super().__init__(*args, **kwargs)
if select is not None:
self.setAttribute("select", select)
[docs]
class HTMLDListElement(HTMLElement):
name = "dl"
[docs]
class HTMLDataElement(HTMLElement):
name = "data"
def __init__(self, *args, value=None, **kwargs):
"""HTMLDataElement
Args:
value (str, optional): Contains the machine-readable value associated with the content.
"""
super().__init__(*args, **kwargs)
if value is not None:
self.setAttribute("value", value)
[docs]
class HTMLDataListElement(HTMLElement):
name = "datalist"
[docs]
class HTMLDialogElement(HTMLElement):
name = "dialog"
def __init__(self, *args, open=None, **kwargs):
"""HTMLDialogElement
Args:
open (bool, optional): Whether the dialog is open or closed.
"""
super().__init__(*args, **kwargs)
if open is not None:
self.setAttribute("open", open)
@property
def open(self) -> bool:
return self.hasAttribute("open")
@open.setter
def open(self, is_open: bool) -> None:
previous = self.open
if is_open:
self.setAttribute("open", True)
else:
self.removeAttribute("open")
if previous != self.open:
self.dispatchEvent(Event("toggle", {"bubbles": False, "cancelable": False}))
def show(self):
self.open = True
return self
def showModal(self):
self.open = True
return self
def close(self, returnValue: Any = ""):
from domonic.events import CloseEvent
self.returnValue = returnValue
self.open = False
self.dispatchEvent(CloseEvent("close", {"bubbles": False, "cancelable": False, "code": 0, "reason": str(returnValue), "wasClean": True}))
return self
[docs]
class HTMLDivElement(HTMLElement):
name = "div"
[docs]
class XMLDocument(Document):
name = "xml"
contentType: str = "application/xml"
[docs]
class HTMLDocument(Document):
name = "html"
contentType: str = "text/html"
[docs]
class HTMLEmbedElement(HTMLElement):
name = "embed"
[docs]
class HTMLFieldSetElement(HTMLElement):
name = "fieldset"
def __init__(self, *args, disabled=None, form=None, name=None, **kwargs):
super().__init__(*args, **kwargs)
if disabled is not None:
self.setAttribute("disabled", disabled)
if form is not None:
self.setAttribute("form", form)
if name is not None:
self.setAttribute("name", name)
[docs]
class HTMLFrameSetElement(HTMLElement):
name = "frameset"
def __init__(self, *args, cols=None, rows=None, **kwargs):
super().__init__(*args, **kwargs)
if cols is not None:
self.setAttribute("cols", cols)
if rows is not None:
self.setAttribute("rows", rows)
[docs]
class HTMLHRElement(HTMLElement):
name = "hr"
[docs]
class HTMLHeadElement(HTMLElement):
name = "head"
[docs]
class HTMLHeadingElement(HTMLElement):
name = "h1"
[docs]
class HTMLIFrameElement(HTMLElement):
name = "iframe"
def __init__(self, *args, src=None, name=None, sandbox=None, allowfullscreen=None, **kwargs):
"""HTMLIFrameElement
Args:
src (str, optional): _description_. Defaults to None.
name (str, optional): _description_. Defaults to None.
sandbox (str, optional): _description_. Defaults to None.
allowfullscreen (str, optional): _description_. Defaults to None.
"""
super().__init__(*args, **kwargs)
if src is not None:
self.setAttribute("src", src)
if name is not None:
self.setAttribute("name", name)
if sandbox is not None:
self.setAttribute("sandbox", sandbox)
if allowfullscreen is not None:
self.setAttribute("allowfullscreen", allowfullscreen)
[docs]
class HTMLImageElement(HTMLElement):
name = "img"
__isempty = True
def __init__(
self,
*args,
alt=None,
src=None,
crossorigin=None,
height=None,
ismap=None,
longdesc=None,
sizes=None,
srcset=None,
usemap=None,
width=None,
**kwargs,
):
"""HTMLImageElement
Args:
alt (str, optional): _description_. Defaults to None.
src (str, optional): _description_. Defaults to None.
crossorigin (str, optional): _description_. Defaults to None.
height (str, optional): _description_. Defaults to None.
ismap (str, optional): _description_. Defaults to None.
longdesc (str, optional): _description_. Defaults to None.
sizes (str, optional): _description_. Defaults to None.
srcset (str, optional): _description_. Defaults to None.
usemap (str, optional): _description_. Defaults to None.
width (str, optional): _description_. Defaults to None.
"""
super().__init__(*args, **kwargs)
if alt is not None:
self.setAttribute("alt", alt)
if src is not None:
self.setAttribute("src", src)
if crossorigin is not None:
self.setAttribute("crossorigin", crossorigin)
if height is not None:
self.setAttribute("height", height)
if ismap is not None:
self.setAttribute("ismap", ismap)
if longdesc is not None:
self.setAttribute("longdesc", longdesc)
if sizes is not None:
self.setAttribute("sizes", sizes)
if srcset is not None:
self.setAttribute("srcset", srcset)
if usemap is not None:
self.setAttribute("usemap", usemap)
if width is not None:
self.setAttribute("width", width)
def load(self):
self.dispatchEvent(Event("loadstart", {"bubbles": False, "cancelable": False}))
self.dispatchEvent(Event("load", {"bubbles": False, "cancelable": False}))
return self
def decode(self) -> bool:
self.dispatchEvent(Event("load", {"bubbles": False, "cancelable": False}))
return True
def error(self):
self.dispatchEvent(Event("error", {"bubbles": False, "cancelable": False}))
return None
def abort(self):
self.dispatchEvent(Event("abort", {"bubbles": False, "cancelable": False}))
return None
[docs]
class HTMLIsIndexElement(HTMLElement):
name = "isindex"
def __init__(self, *args, prompt=None, **kwargs):
super().__init__(*args, **kwargs)
if prompt is not None:
self.setAttribute("prompt", prompt)
[docs]
class HTMLKeygenElement(HTMLElement):
name = "keygen"
__isempty = True
[docs]
class HTMLLIElement(HTMLElement):
name = "li"
[docs]
class HTMLLabelElement(HTMLElement):
name = "label"
# def __init__(self, *args, _for=None, **kwargs):
# """_summary_
# Args:
# _for (_type_, optional): the id of the element that this label is for. Defaults to None.
# """
# super().__init__(*args, **kwargs)
# if _for is not None:
# self.setAttribute('for', _for)
[docs]
class HTMLLegendElement(HTMLElement):
name = "legend"
[docs]
class HTMLLinkElement(HTMLElement):
name = "link"
def __init__(self, *args, rel=None, href=None, type=None, sizes=None, **kwargs):
"""HTMLLinkElement
Args:
rel (str, optional): Specifies the relationship between the current document and the linked resource.
href (str, optional): The URL of the linked resource.
type (str, optional): Specifies the type of the linked resource (like 'text/css').
sizes (str, optional): Defines the sizes of the icons linked.
"""
super().__init__(*args, **kwargs)
if rel is not None:
self.setAttribute("rel", rel)
if href is not None:
self.setAttribute("href", href)
if type is not None:
self.setAttribute("type", type)
if sizes is not None:
self.setAttribute("sizes", sizes)
[docs]
class HTMLMapElement(HTMLElement):
name = "map"
def __init__(self, *args, name=None, **kwargs):
super().__init__(*args, **kwargs)
if name is not None:
self.setAttribute("name", name)
[docs]
class HTMLMeterElement(HTMLElement):
name = "meter"
def __init__(self, *args, value=None, _min=None, _max=None, low=None, high=None, optimum=None, **kwargs):
"""HTMLMeterElement
The <meter> HTML element represents either a scalar value within a known range or a fractional value.
Args:
value (_type_, optional): The current numeric value. This must be between the minimum and maximum values (min attribute and max attribute) if they are specified.
min (_type_, optional): The lower numeric bound of the measured range. This must be less than the maximum value (max attribute), if specified. If unspecified, the minimum value is 0.
max (_type_, optional): The upper numeric bound of the measured range. This must be greater than the minimum value (min attribute), if specified. If unspecified, the maximum value is 1.
low (_type_, optional): _description_. Defaults to None.
high (_type_, optional): _description_. Defaults to None.
optimum (_type_, optional): _description_. Defaults to None.
"""
super().__init__(*args, **kwargs)
if value is not None:
self.setAttribute("value", value)
if _min is not None:
self.setAttribute("_min", _min)
if _max is not None:
self.setAttribute("_max", _max)
if low is not None:
self.setAttribute("low", low)
if high is not None:
self.setAttribute("high", high)
if optimum is not None:
self.setAttribute("optimum", optimum)
[docs]
class HTMLModElement(HTMLElement):
name = "mod"
def __init__(self, *args, datetime=None, **kwargs):
"""HTMLModElement
Args:
datetime (str, optional): The date and time when the modification occurred.
"""
super().__init__(*args, **kwargs)
if datetime is not None:
self.setAttribute("datetime", datetime)
[docs]
class HTMLOListElement(HTMLElement):
name = "ol"
[docs]
class HTMLObjectElement(HTMLElement):
name = "object"
[docs]
class HTMLOptGroupElement(HTMLElement):
name = "optgroup"
[docs]
class HTMLOptionElement(HTMLElement):
name = "option"
def __init__(self, *args, disabled=None, label=None, selected=None, value=None, **kwargs):
"""HTMLOptionElement
Args:
disabled (_type_, optional): _description_. Defaults to None.
label (_type_, optional): _description_. Defaults to None.
selected (_type_, optional): _description_. Defaults to None.
value (_type_, optional): _description_. Defaults to None.
"""
super().__init__(*args, **kwargs)
if disabled is not None:
self.setAttribute("disabled", disabled)
if label is not None:
self.setAttribute("label", label)
if selected is not None:
self.setAttribute("selected", selected)
if value is not None:
self.setAttribute("value", value)
self._default_selected = self.selected
@property
def value(self) -> str:
attr_value = self.getAttribute("value")
return self.textContent if attr_value is None else attr_value
@value.setter
def value(self, new_value: Any) -> None:
self.setAttribute("value", new_value)
@property
def selected(self) -> bool:
return self.hasAttribute("selected")
@selected.setter
def selected(self, is_selected: bool) -> None:
if is_selected:
parent = self.parentNode
if isinstance(parent, HTMLSelectElement) and not parent.hasAttribute("multiple"):
for option in parent.options:
if option is not self:
option.removeAttribute("selected")
self.setAttribute("selected", True)
else:
self.removeAttribute("selected")
@property
def defaultSelected(self) -> bool:
return bool(getattr(self, "_default_selected", self.selected))
@defaultSelected.setter
def defaultSelected(self, is_selected: bool) -> None:
self._default_selected = bool(is_selected)
[docs]
class HTMLOptionsCollection(HTMLCollection):
"""Live collection of a select element's option descendants."""
def __init__(self, select: "HTMLSelectElement"):
super().__init__()
self._select = select
def _options(self) -> list[HTMLOptionElement]:
options: list[HTMLOptionElement] = []
def walk(node):
if not isinstance(node, Element):
return
if node is not self._select and isinstance(node, HTMLOptionElement):
options.append(node)
for child in getattr(node, "childNodes", []):
walk(child)
walk(self._select)
return options
@property
def length(self) -> int:
return len(self._options())
def __len__(self) -> int:
return self.length
def __iter__(self):
return iter(self._options())
def __getitem__(self, index: int | str):
options = self._options()
if isinstance(index, str):
return self.namedItem(index)
return options[index]
[docs]
def item(self, index: int) -> HTMLOptionElement | None:
options = self._options()
return options[index] if 0 <= index < len(options) else None
[docs]
def namedItem(self, name: str) -> HTMLOptionElement | None:
for option in self._options():
if option.getAttribute("id") == name or option.getAttribute("name") == name:
return option
return None
def add(self, element: HTMLOptionElement, before: int | Node | None = None) -> None:
if isinstance(before, int):
reference = self.item(before)
if reference is None:
self._select.appendChild(element)
else:
self._select.insertBefore(element, reference)
return
if isinstance(before, Node):
self._select.insertBefore(element, before)
return
self._select.appendChild(element)
[docs]
def remove(self, index: int) -> None:
option = self.item(index)
if option is not None:
self._select.removeChild(option)
[docs]
class HTMLOutputElement(HTMLElement):
name = "output"
[docs]
class HTMLParagraphElement(HTMLElement):
name = "p"
[docs]
class HTMLParamElement(HTMLElement):
name = "param"
__isempty = True
def __init__(self, *args, name=None, value=None, **kwargs):
"""HTMLParamElement
Args:
name (str, optional): The name of the parameter.
value (str, optional): The value of the parameter.
"""
super().__init__(*args, **kwargs)
if name is not None:
self.setAttribute("name", name)
if value is not None:
self.setAttribute("value", value)
[docs]
class HTMLPictureElement(HTMLElement):
name = "picture"
def __init__(self, *args, **kwargs):
"""HTMLPictureElement
A container for `<source>` elements, allowing the browser to choose from multiple images based on media queries.
"""
super().__init__(*args, **kwargs)
[docs]
class HTMLPreElement(HTMLElement):
name = "pre"
[docs]
class HTMLProgressElement(HTMLElement):
name = "progress"
def __init__(self, *args, value=None, max=None, **kwargs):
"""HTMLProgressElement
Args:
value (str, optional): The current progress value.
max (str, optional): The maximum value of the progress.
"""
super().__init__(*args, **kwargs)
if value is not None:
self.setAttribute("value", value)
if max is not None:
self.setAttribute("max", max)
[docs]
class HTMLQuoteElement(HTMLElement):
name = "q"
def __init__(self, *args, cite=None, **kwargs):
"""HTMLQuoteElement
Args:
cite (str, optional): The source URL for the quotation.
"""
super().__init__(*args, **kwargs)
if cite is not None:
self.setAttribute("cite", cite)
[docs]
class HTMLScriptElement(HTMLElement):
name = "script"
# def __init__(self, *args, _async=None, charset=None, crossorigin=None, defer=None, src=None, type=None, **kwargs):
# """HTMLScriptElement
# Args:
# async (_type_, optional): _description_. Defaults to None.
# charset (_type_, optional): _description_. Defaults to None.
# crossorigin (_type_, optional): _description_. Defaults to None.
# defer (_type_, optional): _description_. Defaults to None.
# src (_type_, optional): _description_. Defaults to None.
# type (_type_, optional): _description_. Defaults to None.
# """
# super().__init__(*args, **kwargs)
# # if _async is not None:
# # self.setAttribute('async', _async)
# if charset is not None:
# self.setAttribute('charset', charset)
# if crossorigin is not None:
# self.setAttribute('crossorigin', crossorigin)
# if defer is not None:
# self.setAttribute('defer', defer)
# if src is not None:
# self.setAttribute('src', src)
# if type is not None:
# self.setAttribute('type', type)
[docs]
class HTMLSelectElement(HTMLElement):
name = "select"
def __init__(
self,
*args,
autofocus: bool = None,
disabled: bool = None,
multiple: bool = None,
name: str = None,
required: bool = None,
size: int = None,
**kwargs,
):
"""HTMLSelectElement
Args:
autofocus (bool, optional): lets you specify that a form control should have input focus when the page loads.
disabled (bool, optional): toggles if user can interact
multiple (bool, optional): If multiple options can be selected in the list.
name (str, optional): This attribute is used to specify the name of the control.
required (bool, optional): indicating that an option with a non-empty string value must be selected.
size (int, optional): the number of rows in the list that should be visible at one time.
"""
super().__init__(*args, **kwargs)
if autofocus is not None:
self.setAttribute("autofocus", autofocus)
if disabled is not None:
self.setAttribute("disabled", disabled)
if multiple is not None:
self.setAttribute("multiple", multiple)
if name is not None:
self.setAttribute("name", name)
if required is not None:
self.setAttribute("required", required)
if size is not None:
self.setAttribute("size", size)
@property
def options(self) -> HTMLOptionsCollection:
return HTMLOptionsCollection(self)
@property
def selectedIndex(self) -> int:
for index, option in enumerate(self.options):
if option.selected:
return index
return -1
@selectedIndex.setter
def selectedIndex(self, index: int) -> None:
for option_index, option in enumerate(self.options):
option.selected = option_index == index
@property
def value(self) -> str:
index = self.selectedIndex
option = self.options.item(index)
return option.value if option is not None else ""
@value.setter
def value(self, new_value: Any) -> None:
string_value = "" if new_value is None else str(new_value)
matched = False
for option in self.options:
is_match = option.value == string_value
option.selected = is_match
matched = matched or is_match
if not matched and not self.hasAttribute("multiple"):
self.selectedIndex = -1
@property
def selectedOptions(self) -> list[HTMLOptionElement]:
return [option for option in self.options if option.selected]
def setValue(self, new_value: Any, *, dispatch_events: bool = True) -> str:
self.value = new_value
if dispatch_events:
_dispatch_value_change_events(self)
return self.value
def selectIndex(self, index: int, *, dispatch_events: bool = True) -> int:
self.selectedIndex = index
if dispatch_events:
_dispatch_value_change_events(self)
return self.selectedIndex
def checkValidity(self) -> bool:
return _is_control_valid(self)
def reportValidity(self) -> bool:
return self.checkValidity()
[docs]
class HTMLShadowElement(HTMLElement):
name = "shadow"
# Currently, the shadow element is obsolete and not typically used in HTML5. Its use was associated with the deprecated Shadow DOM v0 API.
def __init__(self, *args, **kwargs):
"""HTMLShadowElement
The <shadow> element was used in the Shadow DOM v0 specification. It's not commonly used in modern web development.
"""
super().__init__(*args, **kwargs)
[docs]
class HTMLSourceElement(HTMLElement):
name = "source"
__isempty = True
def __init__(self, *args, src=None, type=None, media=None, sizes=None, **kwargs):
"""HTMLSourceElement
Args:
src (str, optional): Specifies the URL of the resource.
type (str, optional): Specifies the MIME type of the resource.
media (str, optional): Specifies the media query for when to apply the source.
sizes (str, optional): Specifies the sizes of the source.
"""
super().__init__(*args, **kwargs)
if src is not None:
self.setAttribute("src", src)
if type is not None:
self.setAttribute("type", type)
if media is not None:
self.setAttribute("media", media)
if sizes is not None:
self.setAttribute("sizes", sizes)
[docs]
class HTMLSpanElement(HTMLElement):
name = "span"
[docs]
class HTMLStyleElement(HTMLElement):
name = "style"
def __init__(self, *args, type=None, media=None, scoped=None, **kwargs):
"""HTMLStyleElement
Args:
type (str, optional): Specifies the type of style sheet.
media (str, optional): Specifies the media for which the styles are intended.
scoped (str, optional): Indicates whether the style is scoped to the element.
"""
super().__init__(*args, **kwargs)
if type is not None:
self.setAttribute("type", type)
if media is not None:
self.setAttribute("media", media)
if scoped is not None:
self.setAttribute("scoped", scoped)
[docs]
class HTMLTableCaptionElement(HTMLElement):
name = "caption"
[docs]
class HTMLTableCellElement(HTMLElement):
name = "td"
[docs]
class HTMLTableColElement(HTMLElement):
name = "col"
__isempty = True
[docs]
class HTMLTableDataCellElement(HTMLElement):
name = "td"
[docs]
class HTMLTableElement(HTMLElement):
name = "table"
def __init__(
self,
*args,
align: str = None,
bgcolor=None,
border=None,
cellpadding=None,
cellspacing=None,
frame=None,
rules=None,
summary=None,
width=None,
**kwargs,
):
"""HTMLTableElement
- in most cases it seems docs are advising to use css instead
Args:
align (str, optional): This enumerated attribute indicates how the table must be aligned inside the containing document.
bgcolor (str, optional): The background color of the table. It is a 6-digit hexadecimal RGB code, prefixed by a '#'. One of the predefined color keywords can also be used.
border (int, optional): The size of the frame surrounding the table. If set to 0, the frame attribute is set to void.
cellpadding (int, optional): This attribute defines the space between the content of a cell and its border, displayed or not. If the cellpadding's length is defined in pixels, this pixel-sized space will be applied to all four sides of the cell's content. If the length is defined using a percentage value, the content will be centered and the total vertical space (top and bottom) will represent this value.
cellspacing (int, optional): This attribute defines the size of the space between two cells in a percentage value or pixels. The attribute is applied both horizontally and vertically, to the space between the top of the table and the cells of the first row, the left of the table and the first column, the right of the table and the last column and the bottom of the table and the last row.
frame (str, optional): This enumerated attribute defines which side of the frame surrounding the table must be displayed.
rules (str, optional): This enumerated attribute defines where rules, i.e. lines, should appear in a table. It can have the following values
summary (str, optional): This attribute defines an alternative text that summarizes the content of the table. Use the <caption> element instead.
width (str, optional): This attribute defines the width of the table. Use the CSS width property instead.
"""
super().__init__(*args, **kwargs)
if align is not None:
self.setAttribute("align", align)
if bgcolor is not None:
self.setAttribute("bgcolor", bgcolor)
if border is not None:
self.setAttribute("border", border)
if cellpadding is not None:
self.setAttribute("cellpadding", cellpadding)
if cellspacing is not None:
self.setAttribute("cellspacing", cellspacing)
if frame is not None:
self.setAttribute("frame", frame)
if rules is not None:
self.setAttribute("rules", rules)
if summary is not None:
self.setAttribute("summary", summary)
if width is not None:
self.setAttribute("width", width)
[docs]
class HTMLTableRowElement(HTMLElement):
name = "tr"
[docs]
class HTMLTableSectionElement(HTMLElement):
name = "tbody"
[docs]
class HTMLDetailsElement(HTMLElement):
name = "details"
def __init__(self, *args, open: bool = None, **kwargs):
super().__init__(*args, **kwargs)
if open is not None:
self.open = open
@property
def open(self) -> bool:
return self.hasAttribute("open")
@open.setter
def open(self, is_open: bool) -> None:
previous = self.open
if is_open:
self.setAttribute("open", True)
else:
self.removeAttribute("open")
if previous != self.open:
self.dispatchEvent(Event("toggle", {"bubbles": False, "cancelable": False}))
def toggle(self) -> bool:
self.open = not self.open
return self.open
[docs]
class HTMLSummaryElement(HTMLElement):
name = "summary"
[docs]
class HTMLSlotElement(HTMLElement):
name = "slot"
def assignedNodes(self, options: dict[str, Any] | None = None) -> list[Node]:
flatten = bool((options or {}).get("flatten"))
root = getattr(self, "parentNode", None)
if not isinstance(root, ShadowRoot):
return []
slot_name = self.getAttribute("name") or ""
assigned: list[Node] = []
for child in getattr(root.host, "childNodes", []):
if isinstance(child, Element):
child_slot = child.getAttribute("slot") or ""
if child_slot == slot_name:
assigned.append(child)
elif slot_name == "":
assigned.append(child)
if assigned:
return assigned
fallback = [child for child in self.childNodes if isinstance(child, Node)]
if flatten:
flattened: list[Node] = []
for child in fallback:
if isinstance(child, HTMLSlotElement):
flattened.extend(child.assignedNodes(options))
else:
flattened.append(child)
return flattened
return fallback
def assignedElements(self, options: dict[str, Any] | None = None) -> list[Element]:
return [node for node in self.assignedNodes(options) if isinstance(node, Element)]
[docs]
class HTMLTemplateElement(HTMLElement):
name = "template"
@property
def content(self):
return DocumentFragment(*self.args)
@content.setter
def content(self, ignore):
self.__content = "".join([each.__str__() for each in self.args])
[docs]
class HTMLTextAreaElement(HTMLElement):
name = "textarea"
def __init__(
self,
*args,
autofocus=None,
cols=None,
disabled=None,
form=None,
maxlength=None,
name=None,
placeholder=None,
readonly=None,
required=None,
rows=None,
wrap=None,
**kwargs,
):
"""HTMLTextAreaElement
Args:
autofocus (_type_, optional): _description_. Defaults to None.
cols (_type_, optional): _description_. Defaults to None.
disabled (_type_, optional): _description_. Defaults to None.
form (_type_, optional): _description_. Defaults to None.
maxlength (_type_, optional): _description_. Defaults to None.
name (_type_, optional): _description_. Defaults to None.
placeholder (_type_, optional): _description_. Defaults to None.
readonly (_type_, optional): _description_. Defaults to None.
required (_type_, optional): _description_. Defaults to None.
rows (_type_, optional): _description_. Defaults to None.
wrap (_type_, optional): _description_. Defaults to None.
"""
super().__init__(*args, **kwargs)
if autofocus is not None:
self.setAttribute("autofocus", autofocus)
if cols is not None:
self.setAttribute("cols", cols)
if disabled is not None:
self.setAttribute("disabled", disabled)
if form is not None:
self.setAttribute("form", form)
if maxlength is not None:
self.setAttribute("maxlength", maxlength)
if name is not None:
self.setAttribute("name", name)
if placeholder is not None:
self.setAttribute("placeholder", placeholder)
if readonly is not None:
self.setAttribute("readonly", readonly)
if required is not None:
self.setAttribute("required", required)
if rows is not None:
self.setAttribute("rows", rows)
if wrap is not None:
self.setAttribute("wrap", wrap)
self._default_value = self.value
@property
def value(self) -> str:
return "" if self.textContent is None else self.textContent
@value.setter
def value(self, new_value: Any) -> None:
self.textContent = "" if new_value is None else str(new_value)
def setValue(self, new_value: Any, *, dispatch_events: bool = True) -> str:
self.value = new_value
if dispatch_events:
_dispatch_value_change_events(self)
return self.value
@property
def defaultValue(self) -> str:
return getattr(self, "_default_value", self.value)
@defaultValue.setter
def defaultValue(self, new_value: Any) -> None:
self._default_value = "" if new_value is None else str(new_value)
def checkValidity(self) -> bool:
return _is_control_valid(self)
def reportValidity(self) -> bool:
return self.checkValidity()
HTMLFormControlsCollection.CONTROL_TYPES = (
HTMLButtonElement,
HTMLFieldSetElement,
HTMLInputElement,
HTMLObjectElement,
HTMLOutputElement,
HTMLSelectElement,
HTMLTextAreaElement,
)
[docs]
class HTMLTimeElement(HTMLElement):
name = "time"
def __init__(self, *args, datetime=None, **kwargs):
"""HTMLTimeElement
Args:
datetime (str, optional): Represents the time value in a machine-readable format.
"""
super().__init__(*args, **kwargs)
if datetime is not None:
self.setAttribute("datetime", datetime)
[docs]
class HTMLTitleElement(HTMLElement):
name = "title"
[docs]
class HTMLTrackElement(HTMLElement):
name = "track"
def __init__(self, *args, kind=None, label=None, src=None, srclang=None, default=None, **kwargs):
"""HTMLTrackElement
Args:
kind (str, optional): Specifies the kind of text track. Can be "subtitles", "captions", "descriptions", or "chapters".
label (str, optional): A user-readable title for the track.
src (str, optional): The URL of the track file.
srclang (str, optional): The language of the track text.
default (bool, optional): Indicates if the track should be shown by default.
"""
super().__init__(*args, **kwargs)
if kind is not None:
self.setAttribute("kind", kind)
if label is not None:
self.setAttribute("label", label)
if src is not None:
self.setAttribute("src", src)
if srclang is not None:
self.setAttribute("srclang", srclang)
if default is not None:
self.setAttribute("default", default)
[docs]
class HTMLUListElement(HTMLElement):
name = "ul"
[docs]
class HTMLUnknownElement(HTMLElement):
name = "unknown"
[docs]
class HTMLVideoElement(HTMLElement):
name = "video"
def __init__(
self,
*args,
autoplay=None,
controls=None,
height=None,
loop=None,
muted=None,
poster=None,
preload=None,
src=None,
width=None,
**kwargs,
):
"""HTMLVideoElement
Args:
autoplay (_type_, optional): _description_. Defaults to None.
controls (_type_, optional): _description_. Defaults to None.
height (_type_, optional): _description_. Defaults to None.
loop (_type_, optional): _description_. Defaults to None.
muted (_type_, optional): _description_. Defaults to None.
poster (_type_, optional): _description_. Defaults to None.
preload (_type_, optional): _description_. Defaults to None.
src (_type_, optional): _description_. Defaults to None.
width (_type_, optional): _description_. Defaults to None.
"""
super().__init__(*args, **kwargs)
if autoplay is not None:
self.setAttribute("autoplay", autoplay)
if controls is not None:
self.setAttribute("controls", controls)
if height is not None:
self.setAttribute("height", height)
if loop is not None:
self.setAttribute("loop", loop)
if muted is not None:
self.setAttribute("muted", muted)
if poster is not None:
self.setAttribute("poster", poster)
if preload is not None:
self.setAttribute("preload", preload)
if src is not None:
self.setAttribute("src", src)
if width is not None:
self.setAttribute("width", width)
[docs]
class HTMLPortalElement(HTMLElement):
name = "portal"
# document can be set manually but will get set each time a new Document is created.
global document
document = Document()
console = Console # legacy. should access via window
# Considered obsolete dom classes ----
# DOMConfiguration - we now use a variation of this name DOMConfig for render settings
# DOMErrorHandler
# DOMImplementationList
# DOMImplementationRegistry
# DOMImplementationSource
# DOMLocator
# DOMObject
# DOMSettableTokenList
# DOMUserData
# ElementTraversal
# Entity
# EntityReference
# NameList
# Notation
# TypeInfo
# UserDataHandler
"""
# self.screen = type('screen', (DOM,), {'name':'screen'})
"""
# https://developer.mozilla.org/en-US/docs/Glossary/Empty_element
# def is_empty(node):
# if its a class,
# if its an instance
# if its a string
# meta = HTMLMetaElement
# br = HTMLBRElement
# img = HTMLImageElement
# input = HTMLInputElement
# param = HTMLParamElement
# source = HTMLSourceElement
# track = HTMLTrackElement
# col = HTMLTableColElement
# keygen = HTMLKeygenElement
# hr = type("hr", (closed_tag, Element), {"name": "hr"})
# wbr = type("wbr", (closed_tag, Element), {"name": "wbr"})
# command = type("command", (closed_tag, Element), {"name": "command"})