Source code for pypdf.actions._actions

"""Action types"""
import sys
from abc import ABC
from enum import Enum, unique
from typing import (
    TYPE_CHECKING,
    cast,
)

from .._utils import logger_warning
from ..errors import ParseError
from ..generic import (
    ArrayObject,
    DictionaryObject,
    NameObject,
    NullObject,
    TextStringObject,
    is_null_or_none,
)

if sys.version_info >= (3, 11):
    from enum import StrEnum
else:
    class StrEnum(str, Enum):
        def __str__(self) -> str:
            return str(self.value)

if TYPE_CHECKING:
    from .._page import PageObject


[docs] @unique class PageTrigger(StrEnum): """Trigger event entries in a page object's additional-actions dictionary.""" OPEN = "open" """A :py:class:`~pypdf.actions.PageTrigger` object triggering an action when the page is opened.""" CLOSE = "close" """A :py:class:`~pypdf.actions.PageTrigger` object triggering an action when the page is closed."""
[docs] class Action(DictionaryObject, ABC): """An action dictionary defines the characteristics and behaviour of an action.""" def __init__(self) -> None: super().__init__() self[NameObject("/Type")] = NameObject("/Action") # The next action or sequence of actions that shall be performed after the action # represented by this dictionary. The value is either a single action dictionary # or an array of action dictionaries that shall be performed in order. self[NameObject("/Next")] = NullObject() # Optional @classmethod def _create_new(cls, page: "PageObject", trigger: PageTrigger, action: "Action") -> None: """ Create a new action and add it to the page. Args: page: The page to add the action. trigger: A :py:class:`~pypdf.actions.PageTrigger` object. action: A :py:class:`~pypdf.actions.Action` object. """ trigger_name = NameObject("/O") if PageTrigger(trigger).value == PageTrigger.OPEN else NameObject("/C") if "/AA" not in page: # Additional actions key not present page[NameObject("/AA")] = DictionaryObject( {trigger_name: action} ) return if isinstance(page["/AA"], NullObject): page[NameObject("/AA")] = DictionaryObject() if not isinstance(page["/AA"].get_object(), DictionaryObject): if page.pdf is not None and getattr(page.pdf, "strict", False): current_type = type(page["/AA"]) raise ParseError( f"The AA entry of the page should be a DictionaryObject. " f"It currently is a {current_type}." ) logger_warning( "The AA entry of the page should be a DictionaryObject. It currently is a %(type)s.", source=__name__, type=type(page["/AA"]) ) return additional_actions = cast(DictionaryObject, page["/AA"]) if is_null_or_none(additional_actions.get(trigger_name)): additional_actions.update({trigger_name: action}) return # The action dictionary's Next entry allows sequences of actions to be # chained together. For example, the effect of clicking a link # annotation with the mouse can be to play a sound, jump to a new # page, and start up a movie. Note that the Next entry is not # restricted to a single action but can contain an array of actions, # each of which in turn can have a Next entry of its own. # §12.6.2 Action dictionaries ISO 32000-2:2020 head = current = additional_actions.get(trigger_name) if not isinstance(head, DictionaryObject): raise ParseError( f"The type in a page object's additional-actions key must be a DictionaryObject: " f"received type {type(head)}" ) current = cast(DictionaryObject, current) visited = set() while True: next_ = current.get("/Next", None) if is_null_or_none(next_): break if not isinstance(next_, (ArrayObject, DictionaryObject)): raise TypeError( f"An action dictionary’s Next entry must be an Action dictionary " f"or an array of Action dictionaries: received type {type(next_)}" ) id_ = id(next_) if id_ in visited: logger_warning("Detected cycle in the action tree for %(current)s", source=__name__, current=current) break visited.add(id_) if isinstance(next_, ArrayObject): current = next_[-1] else: current = next_ if not is_null_or_none(next_ := current.get("/Next")) and id(next_) in visited: logger_warning("Detected cycle in the action tree for %(current)s", source=__name__, current=current) current[NameObject("/Next")] = action additional_actions.update({trigger_name: head}) @classmethod def _delete(cls, page: "PageObject", trigger: PageTrigger) -> None: """ Delete an object on the page. Args: page: The page to add the action. trigger: A :py:class:`~pypdf.actions.PageTrigger` object. """ if "/AA" not in page: return trigger_name = NameObject("/O") if PageTrigger(trigger).value == PageTrigger.OPEN else NameObject("/C") additional_actions = cast(DictionaryObject, page["/AA"]) if trigger_name not in additional_actions: return del additional_actions[trigger_name] if not additional_actions: del page["/AA"]
[docs] class JavaScript(Action): """ Upon invocation of an ECMAScript action, a PDF processor shall execute a script that is written in the ECMAScript programming language. ECMAScript extensions described in ISO/DIS 21757-1 shall also be allowed. """ def __init__(self, js: str) -> None: """ Initialize JavaScript with a string. Args: js: A text string containing the ECMAScript script to be executed. """ super().__init__() self[NameObject("/S")] = NameObject("/JavaScript") self[NameObject("/JS")] = TextStringObject(js)