Source code for ansys.dyna.core.lib.option_card

# Copyright (C) 2021 - 2024 ANSYS, Inc. and/or its affiliates.
# SPDX-License-Identifier: MIT
#
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
# in the Software without restriction, including without limitation the rights
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
# copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in all
# copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
# SOFTWARE.

import abc
import io
import typing

from ansys.dyna.core.lib.card_interface import CardInterface
from ansys.dyna.core.lib.card_writer import write_cards
from ansys.dyna.core.lib.format_type import format_type
from ansys.dyna.core.lib.io_utils import write_or_return
from ansys.dyna.core.lib.parameters import ParameterSet


[docs] class OptionSpec: def __init__(self, name: str, card_order: int, title_order: int): self._name = name self._card_order = card_order self._title_order = title_order @property
[docs] def name(self) -> str: return self._name
@name.setter def name(self, value: str) -> None: self._name = value @property
[docs] def card_order(self) -> int: return self._card_order
@card_order.setter def card_order(self, value: int) -> None: self._card_order = value @property
[docs] def title_order(self) -> int: return self._title_order
@title_order.setter def title_order(self, value: int) -> None: self._title_order = value
[docs] def __repr__(self) -> str: return f"OptionSpec(name={self.name}, card_order={self.card_order}, title_order={self.title_order})"
[docs] class OptionCardSet(CardInterface): def __init__( self, option_spec: OptionSpec, cards: typing.List[CardInterface], **kwargs, ): self._keyword: typing.Any = kwargs.get("keyword", None) self._option_spec = option_spec self._cards: typing.List[CardInterface] = cards self._format_type: format_type = kwargs.get("format", format_type.default) @property
[docs] def cards(self) -> typing.List[CardInterface]: return self._cards
@property
[docs] def option_spec(self) -> OptionSpec: return self._option_spec
@property
[docs] def name(self) -> str: return self._option_spec.name
@property
[docs] def title_order(self) -> int: return self._option_spec.title_order
@property
[docs] def card_order(self) -> int: return self._option_spec.card_order
@property
[docs] def active(self) -> bool: return self._keyword.is_option_active(self.name)
@active.setter def active(self, value: bool) -> None: if value: self._keyword.activate_option(self.name) else: self._keyword.deactivate_option(self.name) @property
[docs] def format(self) -> format_type: """Get the card format type.""" return self._format_type
@format.setter def format(self, value: format_type) -> None: """Set the card format type.""" self._format_type = value
[docs] def __hash__(self): return hash(self.card_order)
[docs] def __lt__(self, other: "OptionCardSet"): return self.card_order < other.card_order
[docs] def read(self, buf: typing.TextIO, parameter_set: ParameterSet = None) -> bool: """Read from buf.""" for card in self._cards: card.read(buf, parameter_set)
[docs] def write( self, format: typing.Optional[format_type] = None, buf: typing.Optional[typing.TextIO] = None, comment: typing.Optional[bool] = True, ) -> typing.Union[str, None]: """Renders the card in the dyna keyword format. :param buf: Buffer to write to. If None, the output is returned as a string :param format: format_type to use. Default to standard. """ def _write(buf): # TODO - write_cards should check the active func if self.active: write_cards(self._cards, buf, format, comment) return write_or_return(buf, _write)
[docs] class OptionsInterface(metaclass=abc.ABCMeta): """Abstract base class for option card api interface.""" @abc.abstractmethod
[docs] def get_option_spec(self, name: str) -> OptionSpec: raise NotImplementedError
@abc.abstractmethod
[docs] def deactivate_option(self, name: str) -> None: raise NotImplementedError
@abc.abstractmethod
[docs] def activate_option(self, name: str) -> None: raise NotImplementedError
@abc.abstractmethod
[docs] def is_option_active(self, name: str) -> bool: raise NotImplementedError
@property @abc.abstractmethod
[docs] def option_specs(self) -> typing.Iterable[OptionSpec]: """Get the card format type.""" raise NotImplementedError
[docs] class OptionAPI: """API for an individual option associated with a keyword.""" def __init__(self, options_api: OptionsInterface, name: str): self._options_api = options_api self._name = name @property
[docs] def active(self) -> bool: return self._options_api.is_option_active(self._name)
@active.setter def active(self, value: bool) -> None: option_spec: OptionSpec = self._options_api.get_option_spec(self._name) if value: self._options_api.activate_option(self._name) # deactivate all other options with the same card order and title order # since they will be mutually exclusive for any_option_spec in self._options_api.option_specs: if any_option_spec.name == self._name: continue if ( any_option_spec.title_order == option_spec.title_order and any_option_spec.card_order == option_spec.card_order ): self._options_api.deactivate_option(any_option_spec.name) else: self._options_api.deactivate_option(self._name)
[docs] class Options: """Option collection associated with an options API.""" def __init__(self, api: OptionsInterface): self._api = api
[docs] def __getitem__(self, name: str) -> OptionAPI: """Gets the option with the given name.""" return OptionAPI(self._api, name)
[docs] def __repr__(self) -> str: option_specs = self._api.option_specs if len(option_specs) == 0: return "" sio = io.StringIO() sio.write("Options:") for option_spec in option_specs: active = self._api.is_option_active(option_spec.name) active_string = "active" if active else "not active" sio.write(f"\n {option_spec.name} option is {active_string}.") return sio.getvalue()
@property
[docs] def api(self) -> OptionsInterface: return self._api