Source code for pyNastran.bdf.cards.bdf_sets

"""
All set cards are defined in this file.  This includes:

* sets
  * SET1, SET2, SET3, RADSET # ??? RADSET
* asets - aset, aset1
* omits - omit, omit1
* bsets - bset, bset1
* csets - cset, cset1
* qsets - qset, qset1
* usets - uset, uset1  # USET 1 is not supported

The superelement sets start with SE:
* se_bsets - sebset, sebset1
* se_csets - secset, secset1
* se_qsets - seqset, seqset1
* se_usets - seuset, seuset1
*se_sets
  * SESET
  * SEQSEP

 #* Set
 #* SetSuper

+------------+-----------------+
| Entry Type | Equivalent Type |
+============+=================+
|  SEQSETi   | QSETi           |
+------------+-----------------+
|  SESUP     | SUPORT          |
+------------+-----------------+
|  SECSETi   | CSETi           |
+------------+-----------------+
|  SEBSETi   | BSETi           |
+------------+-----------------+

"""
from __future__ import annotations
from typing import Union, Optional, Any, TYPE_CHECKING
import numpy as np

from pyNastran.utils.numpy_utils import integer_types, integer_string_types
from pyNastran.bdf.cards.base_card import (
    BaseCard, _node_ids, expand_thru, write_card
)
from pyNastran.bdf.cards.collpase_card import collapse_thru, condense, build_thru_packs
from pyNastran.bdf.field_writer_8 import print_card_8
from pyNastran.bdf.field_writer_16 import print_card_16
from pyNastran.bdf.bdf_interface.assign_type import (
    integer, double, double_or_blank,
    integer_or_blank, integer_or_string,
    parse_components, parse_components_or_blank, components_or_blank as fcomponents_or_blank,
    fields, string, integer_string_or_blank,
)
if TYPE_CHECKING:  # pragma: no cover
    from pyNastran.bdf.bdf import BDF


[docs] class Set(BaseCard): """Generic Class all SETx cards inherit from""" def __init__(self): #: list of IDs in the SETx self.ids = []
[docs] def clean_ids(self) -> None: """eliminates duplicate IDs from self.IDs and sorts self.IDs""" self.ids = list(set(self.ids)) self.ids.sort()
[docs] def repr_fields(self)-> list[Optional[Union[int, float, str]]]: list_fields = self.raw_fields() return list_fields
def __repr__(self) -> str: try: out = self.comment + print_card_8(self.repr_fields()) except RuntimeError: out = self.comment + print_card_16(self.repr_fields()) return out
[docs] def write_card(self, size: int=8, is_double: bool=False) -> str: card = self.repr_fields() return write_card(self.comment, card, size, is_double)
[docs] class SetSuper(Set): """Generic Class all Superelement SETx cards inherit from.""" def __init__(self): Set.__init__(self) #: Superelement identification number. Must be a primary superelement. #: (Integer >= 0) self.seid = None #: list of IDs in the SESETx self.ids = None
[docs] class ABCQSet(Set): """ Generic Class ASET, BSET, CSET, QSET cards inherit from. Defines degrees-of-freedom in the analysis set (A-set) +------+-----+----+-----+------+-----+----+-----+----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +======+=====+====+=====+======+=====+====+=====+====+ | ASET | ID1 | C1 | ID2 | C2 | ID3 | C3 | ID4 | C4 | +------+-----+----+-----+------+-----+----+-----+----+ | ASET | 16 | 2 | 23 | 3516 | 1 | 4 | | | +------+-----+----+-----+------+-----+----+-----+----+ """ type = 'ABCQSet'
[docs] def _finalize_hdf5(self, encoding): """hdf5 helper function""" if isinstance(self.ids, np.ndarray): self.ids = self.ids.tolist() if isinstance(self.components, np.ndarray): self.components = self.components.tolist()
def __init__(self, ids: list[int], components: list[int], comment: str='') -> None: Set.__init__(self) if comment: self.comment = comment #: Identifiers of grids points. (Integer > 0) if isinstance(ids, int): ids = [ids] self.ids = ids self.components = components self.ids_ref = None
[docs] def validate(self) -> None: assert isinstance(self.ids, list), type(self.ids) assert isinstance(self.components, list), type(self.components) assert len(self.ids) == len(self.components), 'len(ids)=%s len(components)=%s' % (len(self.ids), len(self.components))
[docs] @classmethod def add_card(cls, card, comment=''): ids = [] components = [] nterms = len(card) // 2 for n in range(nterms): i = n * 2 + 1 idi = integer(card, i, 'ID' + str(n)) component = parse_components_or_blank(card, i + 1, 'component' + str(n)) ids.append(idi) components.append(component) return cls(ids, components, comment=comment)
@classmethod def add_op2_data(cls, data: list[Any], comment: str='') -> ABCQSet: ids = [data[0]] components = [data[1]] return cls(ids, components, comment=comment)
[docs] def cross_reference(self, model: BDF) -> None: """ Cross links the card so referenced cards can be extracted directly Parameters ---------- model : BDF() the BDF object """ msg = ', which is required by %s' % self.type self.ids_ref = model.EmptyNodes(self.node_ids, msg=msg)
[docs] def uncross_reference(self) -> None: """Removes cross-reference links""" self.ids = self.node_ids self.ids_ref = None
@property def node_ids(self) -> list[int]: if self.ids_ref is None: return self.ids msg = ', which is required by %s' % self.type nids = _node_ids(self, self.ids, allow_empty_nodes=True, msg=msg) return nids
[docs] def raw_fields(self): """gets the "raw" card without any processing as a list for printing""" list_fields = [self.type] # ASET, BSET for (idi, comp) in zip(self.node_ids, self.components): list_fields += [idi, comp] return list_fields
def __repr__(self): list_fields = self.raw_fields() return self.comment + print_card_8(list_fields)
[docs] class SuperABCQSet(Set): """ Generic Class ASET, BSET, CSET, QSET cards inherit from. Defines degrees-of-freedom in the analysis set (A-set) +--------+------+-----+----+-----+------+-----+-----+-----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +========+======+=====+====+=====+======+=====+=====+=====+ | SEBSET | SEID | ID1 | C1 | ID2 | C2 | ID3 | C3 | | +--------+------+-----+----+-----+------+-----+-----+-----+ | SEBSET | 100 | 16 | 2 | 23 | 3516 | 1 | 4 | | +--------+------+-----+----+-----+------+-----+-----+-----+ """ type = 'SuperABCQSet'
[docs] def _finalize_hdf5(self, encoding): """hdf5 helper function""" if isinstance(self.ids, np.ndarray): self.ids = self.ids.tolist() if isinstance(self.components, np.ndarray): self.components = self.components.tolist()
def __init__(self, seid, ids, components, comment=''): Set.__init__(self) if comment: self.comment = comment self.seid = seid #: Identifiers of grids points. (Integer > 0) self.ids = ids self.components = components self.ids_ref = None
[docs] def validate(self): assert isinstance(self.ids, list), type(self.ids) assert isinstance(self.components, list), type(self.components) assert len(self.ids) == len(self.components), 'len(ids)=%s len(components)=%s' % (len(self.ids), len(self.components))
[docs] @classmethod def add_card(cls, card, comment=''): seid = integer(card, 1, 'seid') ids = [] components = [] nfields = len(card) nterms = nfields // 2 - 1 delta = nfields % 2 assert delta == 0, 'The number of fields must be even; nfields=%s\ncard=%s' % (nfields, card) for n in range(nterms): i = n * 2 + 2 idi = integer(card, i, 'ID' + str(n)) component = parse_components_or_blank(card, i + 1, 'component' + str(n)) ids.append(idi) components.append(component) return cls(seid, ids, components, comment=comment)
[docs] def cross_reference(self, model: BDF) -> None: """ Cross links the card so referenced cards can be extracted directly Parameters ---------- model : BDF() the BDF object """ msg = ', which is required by %s seid=%s' % (self.type, self.seid) self.ids_ref = model.EmptyNodes(self.node_ids, msg=msg)
[docs] def uncross_reference(self) -> None: """Removes cross-reference links""" self.ids = self.node_ids self.ids_ref = None
@property def node_ids(self): msg = ', which is required by %s seid=%s' % (self.type, self.seid) if self.ids_ref is None: return self.ids return _node_ids(self, self.ids_ref, allow_empty_nodes=True, msg=msg)
[docs] def raw_fields(self): """gets the "raw" card without any processing as a list for printing""" list_fields = [self.type, self.seid] # SEASET, SEBSET for (idi, comp) in zip(self.node_ids, self.components): list_fields += [idi, comp] return list_fields
def __repr__(self): list_fields = self.raw_fields() return self.comment + print_card_8(list_fields)
[docs] class ASET(ABCQSet): """ Defines degrees-of-freedom in the analysis set (A-set). +------+-----+----+-----+------+-----+----+-----+----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +======+=====+====+=====+======+=====+====+=====+====+ | ASET | ID1 | C1 | ID2 | C2 | ID3 | C3 | ID4 | C4 | +------+-----+----+-----+------+-----+----+-----+----+ | ASET | 16 | 2 | 23 | 3516 | 1 | 4 | | | +------+-----+----+-----+------+-----+----+-----+----+ """ type = 'ASET' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): ids = [1, 2] components = ['123', '456'] return ASET(ids, components, comment='')
def __init__(self, ids, components, comment=''): """ Creates an ASET card, which defines the degree of freedoms that will be retained during an ASET modal reduction. Parameters ---------- ids : list[int] the GRID/SPOINT ids components : list[str] the degree of freedoms to be retained (e.g., '1', '123') comment : str; default='' a comment for the card ..note :: the length of components and ids must be the same """ ABCQSet.__init__(self, ids, components, comment)
[docs] class BSET(ABCQSet): """ Defines analysis set (a-set) degrees-of-freedom to be fixed (b-set) during generalized dynamic reduction or component mode synthesis calculations. +------+-----+----+-----+------+-----+----+-----+----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +======+=====+====+=====+======+=====+====+=====+====+ | BSET | ID1 | C1 | ID2 | C2 | ID3 | C3 | ID4 | C4 | +------+-----+----+-----+------+-----+----+-----+----+ | BSET | 16 | 2 | 23 | 3516 | 1 | 4 | | | +------+-----+----+-----+------+-----+----+-----+----+ """ type = 'BSET' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): ids = [1, 2] components = ['123', '456'] return BSET(ids, components, comment='')
def __init__(self, ids, components, comment=''): """ Creates an BSET card, which defines the degree of freedoms that will be fixed during a generalized dynamic reduction or component model synthesis calculation. Parameters ---------- ids : list[int] the GRID/SPOINT ids components : list[str] the degree of freedoms to be fixed (e.g., '1', '123') comment : str; default='' a comment for the card ..note :: the length of components and ids must be the same """ ABCQSet.__init__(self, ids, components, comment)
[docs] class CSET(ABCQSet): """ Defines the degree of freedoms that will be free during a generalized dynamic reduction or component model synthesis calculation. +------+-----+----+-----+------+-----+----+-----+----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +======+=====+====+=====+======+=====+====+=====+====+ | CSET | ID1 | C1 | ID2 | C2 | ID3 | C3 | ID4 | C4 | +------+-----+----+-----+------+-----+----+-----+----+ | CSET | 16 | 2 | 23 | 3516 | 1 | 4 | | | +------+-----+----+-----+------+-----+----+-----+----+ """ type = 'CSET' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): ids = [1, 2] components = ['123', '456'] return CSET(ids, components, comment='')
def __init__(self, ids, components, comment=''): """ Creates an CSET card, which defines the degree of freedoms that will be free during a generalized dynamic reduction or component model synthesis calculation. Parameters ---------- ids : list[int] the GRID/SPOINT ids components : list[str] the degree of freedoms to be free (e.g., '1', '123') comment : str; default='' a comment for the card ..note :: the length of components and ids must be the same """ ABCQSet.__init__(self, ids, components, comment)
[docs] class QSET(ABCQSet): """ Defines generalized degrees-of-freedom (q-set) to be used for dynamic reduction or component mode synthesis. +------+-----+----+-----+------+-----+----+-----+----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +======+=====+====+=====+======+=====+====+=====+====+ | QSET | ID1 | C1 | ID2 | C2 | ID3 | C3 | ID4 | C4 | +------+-----+----+-----+------+-----+----+-----+----+ | QSET | 16 | 2 | 23 | 3516 | 1 | 4 | | | +------+-----+----+-----+------+-----+----+-----+----+ """ type = 'QSET' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): ids = [1, 2] components = ['123', '456'] return QSET(ids, components, comment='')
def __init__(self, ids, components, comment=''): """ Creates a QSET card, which defines generalized degrees of freedom (q-set) to be used for dynamic reduction or component mode synthesis. Parameters ---------- ids : list[int] the GRID/SPOINT ids components : list[str] the degree of freedoms to be created (e.g., '1', '123') comment : str; default='' a comment for the card """ ABCQSet.__init__(self, ids, components, comment)
[docs] class ABCQSet1(Set): """ Generic Class ASET1, BSET1, QSET1 cards inherit from. Defines degrees-of-freedom in the analysis set (a-set). +-------+-----+-----+------+-----+-----+-----+-----+-----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +=======+=====+=====+======+=====+=====+=====+=====+=====+ | xSET1 | C | ID1 | ID2 | ID3 | ID4 | ID5 | ID6 | ID7 | +-------+-----+-----+------+-----+-----+-----+-----+-----+ | | ID8 | ID9 | | | | | | | +-------+-----+-----+------+-----+-----+-----+-----+-----+ | xSET1 | C | ID1 | THRU | ID2 | | | | | +-------+-----+-----+------+-----+-----+-----+-----+-----+ """ type = 'ABQSet1'
[docs] def _finalize_hdf5(self, encoding): """hdf5 helper function""" if isinstance(self.ids, np.ndarray): self.ids = self.ids.tolist()
def __init__(self, ids, components, comment=''): Set.__init__(self) if comment: self.comment = comment #: Component number. (Integer zero or blank for scalar points or any #: unique combination of the Integers 1 through 6 for grid points with #: no embedded blanks.) self.components = components #: Identifiers of grids points. (Integer > 0) self.ids = expand_thru(ids) self.ids_ref = None self.use_thru = True
[docs] @classmethod def add_card(cls, card, comment=''): components = fcomponents_or_blank(card, 1, 'components', '0') nfields = len(card) ids = [] i = 1 for ifield in range(2, nfields): idi = integer_string_or_blank(card, ifield, 'ID%i' % i) if idi: i += 1 ids.append(idi) return cls(ids, components, comment=comment)
@classmethod def add_op2_data(cls, data, comment=''): """[123456 0 1 101]""" components = str(data[0]) thru_flag = data[1] if thru_flag == 0: ids = data[2:] elif thru_flag == 1: assert len(data) == 4, data #ids = [data[2], 'THRU', data[3]] ids = list(range(data[2], data[3]+1)) else: raise NotImplementedError('thru_flag=%s data=%s' % (thru_flag, data)) return cls(ids, components, comment=comment)
[docs] def cross_reference(self, model: BDF) -> None: """ Cross links the card so referenced cards can be extracted directly Parameters ---------- model : BDF() the BDF object """ msg = ', which is required by %s' % self.type self.ids_ref = model.EmptyNodes(self.node_ids, msg=msg)
[docs] def uncross_reference(self) -> None: """Removes cross-reference links""" self.ids = self.node_ids self.ids_ref = None
#@property #def node_ids(self): #return self.get_ids() @property def node_ids(self): msg = ', which is required by %s' % self.type if self.ids_ref is None: return self.ids return _node_ids(self, self.ids_ref, allow_empty_nodes=True, msg=msg)
[docs] def raw_fields(self): """gets the "raw" card without any processing as a list for printing""" if self.use_thru: node_ids_list = collapse_thru(self.node_ids) else: node_ids_list = self.node_ids list_fields = [self.type, self.components] + node_ids_list return list_fields
def __repr__(self): list_fields = self.raw_fields() return self.comment + print_card_8(list_fields)
[docs] class SuperABCQSet1(Set): """ Generic Class SEBSET1, SEQSET1 cards inherit from. Defines degrees-of-freedom in the analysis set (a-set). +----------+------+-----+------+------+-----+-----+-----+-----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +==========+======+=====+======+======+=====+=====+=====+=====+ | SEBSET1 | SEID | C | ID1 | ID2 | ID3 | ID4 | ID5 | ID6 | +----------+------+-----+------+------+-----+-----+-----+-----+ | | ID7 | ID9 | | | | | | | +----------+------+-----+------+------+-----+-----+-----+-----+ | SEBSET1 | SEID | C | ID1 | THRU | ID2 | | | | +----------+------+-----+------+------+-----+-----+-----+-----+ """ type = 'SuperABCQSet1'
[docs] def _finalize_hdf5(self, encoding): """hdf5 helper function""" if isinstance(self.ids, np.ndarray): self.ids = self.ids.tolist()
def __init__(self, seid, ids, components, comment=''): Set.__init__(self) if comment: self.comment = comment self.seid = seid #: Component number. (Integer zero or blank for scalar points or any #: unique combination of the Integers 1 through 6 for grid points with #: no embedded blanks.) self.components = components #: Identifiers of grids points. (Integer > 0) self.ids = expand_thru(ids) #print('ids =', self.ids) assert None not in self.ids self.ids_ref = None self.validate()
[docs] def validate(self): if not isinstance(self.components, integer_string_types): msg = 'type(components)=%s must be an int/string' % type(self.components) raise TypeError(msg)
[docs] @classmethod def add_card(cls, card, comment=''): seid = integer(card, 1, 'seid') components = fcomponents_or_blank(card, 2, 'components', '0') nfields = len(card) ids = [] i = 1 for ifield in range(3, nfields): idi = integer_string_or_blank(card, ifield, 'ID%d' % i) if idi: i += 1 ids.append(idi) ids = expand_thru(ids) return cls(seid, ids, components, comment=comment)
@classmethod def add_op2_data(cls, data, comment=''): seid, components, nids = data #assert None not in components, 'Type=%s components=%s' % (cls.type, components) assert None not in nids, 'Type=%s nids=%s' % (cls.type, nids) assert -1 not in nids, 'nids=%s' % (nids.tolist()) assert 0 not in nids, 'nids=%s' % (nids.tolist()) return cls(seid, nids, components, comment=comment)
[docs] def cross_reference(self, model: BDF) -> None: """ Cross links the card so referenced cards can be extracted directly Parameters ---------- model : BDF() the BDF object """ msg = ', which is required by %s seid=%s' % (self.type, self.seid) self.ids_ref = model.EmptyNodes(self.node_ids, msg=msg)
[docs] def uncross_reference(self) -> None: """Removes cross-reference links""" self.ids = self.node_ids self.ids_ref = None
@property def node_ids(self): msg = ', which is required by %s seid=%s' % (self.type, self.seid) if self.ids_ref is None: return self.ids return _node_ids(self, self.ids_ref, allow_empty_nodes=True, msg=msg)
[docs] def raw_fields(self): """gets the "raw" card without any processing as a list for printing""" list_fields = [self.type, self.seid, self.components] + collapse_thru(self.node_ids) return list_fields
def __repr__(self): list_fields = self.raw_fields() return self.comment + print_card_8(list_fields)
[docs] class ASET1(ABCQSet1): """ Defines degrees-of-freedom in the analysis set (a-set) +-------+-----+-----+------+-----+-----+-----+-----+-----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +=======+=====+=====+======+=====+=====+=====+=====+=====+ | ASET1 | C | ID1 | ID2 | ID3 | ID4 | ID5 | ID6 | ID7 | +-------+-----+-----+------+-----+-----+-----+-----+-----+ | | ID8 | ID9 | | | | | | | +-------+-----+-----+------+-----+-----+-----+-----+-----+ | ASET1 | C | ID1 | THRU | ID2 | | | | | +-------+-----+-----+------+-----+-----+-----+-----+-----+ """ type = 'ASET1' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): ids = [1, 2] components = '123' return ASET1(ids, components, comment='')
def __init__(self, ids, components, comment=''): """ Creates an ASET1 card, which defines the degree of freedoms that will be retained during an ASET modal reduction. Parameters ---------- ids : list[int] the GRID/SPOINT ids components : str the degree of freedoms to be retained (e.g., '1', '123') comment : str; default='' a comment for the card """ ABCQSet1.__init__(self, ids, components, comment)
[docs] class OMIT(ABCQSet): """ Defines analysis set (a-set) degrees-of-freedom to be fixed (b-set) during generalized dynamic reduction or component mode synthesis calculations. +------+-----+----+-----+------+-----+----+-----+----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +======+=====+====+=====+======+=====+====+=====+====+ | OMIT | ID1 | C1 | ID2 | C2 | ID3 | C3 | ID4 | C4 | +------+-----+----+-----+------+-----+----+-----+----+ | OMIT | 16 | 2 | 23 | 3516 | 1 | 4 | | | +------+-----+----+-----+------+-----+----+-----+----+ """ type = 'OMIT' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): ids = [1, 2] components = ['123', '456'] return BSET(ids, components, comment='')
def __init__(self, ids, components, comment=''): """ Creates an BSET card, which defines the degree of freedoms that will be fixed during a generalized dynamic reduction or component model synthesis calculation. Parameters ---------- ids : list[int] the GRID/SPOINT ids components : list[str] the degree of freedoms to be fixed (e.g., '1', '123') comment : str; default='' a comment for the card ..note :: the length of components and ids must be the same """ ABCQSet.__init__(self, ids, components, comment)
[docs] class OMIT1(ABCQSet1): """ Defines degrees-of-freedom to be excluded (o-set) from the analysis set (a-set). +-------+-----+-----+------+-----+-----+-----+-----+-----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +=======+=====+=====+======+=====+=====+=====+=====+=====+ | OMIT | C | ID1 | ID2 | ID3 | ID4 | ID5 | ID6 | ID7 | +-------+-----+-----+------+-----+-----+-----+-----+-----+ | | ID8 | ID9 | | | | | | | +-------+-----+-----+------+-----+-----+-----+-----+-----+ | OMIT1 | C | ID1 | THRU | ID2 | | | | | +-------+-----+-----+------+-----+-----+-----+-----+-----+ """ type = 'OMIT1' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): ids = [1, 2] components = '123' return OMIT1(ids, components, comment='')
def __init__(self, ids, components, comment=''): """ Creates an OMIT1 card, which defines the degree of freedoms that will be excluded (o-set) from the analysis set (a-set). Parameters ---------- ids : list[int] the GRID/SPOINT ids components : str the degree of freedoms to be omitted (e.g., '1', '123') comment : str; default='' a comment for the card """ ABCQSet1.__init__(self, ids, components, comment)
[docs] class BSET1(ABCQSet1): """ Defines analysis set (a-set) degrees-of-freedom to be fixed (b-set) during generalized dynamic reduction or component mode synthesis calculations. +-------+-----+-----+------+-----+-----+-----+-----+-----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +=======+=====+=====+======+=====+=====+=====+=====+=====+ | BSET1 | C | ID1 | ID2 | ID3 | ID4 | ID5 | ID6 | ID7 | +-------+-----+-----+------+-----+-----+-----+-----+-----+ | | ID8 | ID9 | | | | | | | +-------+-----+-----+------+-----+-----+-----+-----+-----+ | BSET1 | C | ID1 | THRU | ID2 | | | | | +-------+-----+-----+------+-----+-----+-----+-----+-----+ """ type = 'BSET1' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): ids = [1, 2] components = '123' return BSET1(ids, components, comment='')
def __init__(self, ids, components, comment=''): """ Creates an BSET1 card, which defines the degree of freedoms that will be fixed during a generalized dynamic reduction or component model synthesis calculation. Parameters ---------- ids : list[int] the GRID/SPOINT ids components : str the degree of freedoms to be fixed (e.g., '1', '123') comment : str; default='' a comment for the card """ ABCQSet1.__init__(self, ids, components, comment)
[docs] class CSET1(ABCQSet1): """ Defines the degree of freedoms that will be free during a generalized dynamic reduction or component model synthesis calculation. +-------+-----+-----+------+-----+-----+-----+-----+-----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +=======+=====+=====+======+=====+=====+=====+=====+=====+ | CSET1 | C | ID1 | ID2 | ID3 | ID4 | ID5 | ID6 | ID7 | +-------+-----+-----+------+-----+-----+-----+-----+-----+ | | ID8 | ID9 | | | | | | | +-------+-----+-----+------+-----+-----+-----+-----+-----+ | CSET1 | C | ID1 | THRU | ID2 | | | | | +-------+-----+-----+------+-----+-----+-----+-----+-----+ | CSET1 | ,, | ALL | | | | | | | +-------+-----+-----+------+-----+-----+-----+-----+-----+ """ type = 'CSET1' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): ids = [1, 2] components = '123' return CSET1(ids, components, comment='')
def __init__(self, ids, components, comment=''): """ Creates an CSET1 card, which defines the degree of freedoms that will be free during a generalized dynamic reduction or component model synthesis calculation. Parameters ---------- ids : list[int] the GRID/SPOINT ids components : str the degree of freedoms to be free (e.g., '1', '123') comment : str; default='' a comment for the card """ Set.__init__(self) if comment: self.comment = comment #: Identifiers of grids points. (Integer > 0) self.ids = expand_thru(ids) self.components = components self.ids_ref = None
[docs] @classmethod def add_card(cls, card, comment=''): """ Adds a CSET1 card from ``BDF.add_card(...)`` Parameters ---------- card : BDFCard() a BDFCard object comment : str; default='' a comment for the card """ if integer_string_or_blank(card, 2, 'C') == 'ALL': components = '123456' else: components = parse_components(card, 1, 'components') ids = [] id_count = 1 for ifield in range(2, len(card)): idi = integer_or_string(card, ifield, 'ID%i' % id_count) ids.append(idi) id_count += 1 return CSET1(ids, components, comment=comment)
#def cross_reference(self, model: BDF) -> None: #""" #Cross links the card so referenced cards can be extracted directly #Parameters #---------- #model : BDF() #the BDF object #""" #msg = ', which is required by CSET1' #self.ids_ref = model.EmptyNodes(self.node_ids, msg=msg) #def uncross_reference(self) -> None: #"""Removes cross-reference links""" #self.ids = self.node_ids #self.ids_ref = None #@property #def node_ids(self): #msg = ', which is required by CSET1' #if self.ids_ref is None: #return self.ids #return _node_ids(self, self.ids_ref, allow_empty_nodes=True, msg=msg)
[docs] def raw_fields(self): """gets the "raw" card without any processing as a list for printing""" list_fields = ['CSET1', self.components] + collapse_thru(self.node_ids) return list_fields
def __repr__(self): list_fields = self.raw_fields() return self.comment + print_card_8(list_fields)
[docs] class QSET1(ABCQSet1): """ Defines generalized degrees-of-freedom (q-set) to be used for dynamic reduction or component mode synthesis. """ type = 'QSET1' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): ids = [1, 2] components = '123' return QSET1(ids, components, comment='')
def __init__(self, ids, components, comment=''): """ Creates a QSET1 card, which defines generalized degrees of freedom (q-set) to be used for dynamic reduction or component mode synthesis. Parameters ---------- ids : list[int] the GRID/SPOINT ids components : str the degree of freedoms to be created (e.g., '1', '123') comment : str; default='' a comment for the card """ ABCQSet1.__init__(self, ids, components, comment)
[docs] class SET1(Set): """ Defines a list of structural grid points or element identification numbers. +------+--------+--------+-----+------+-----+-----+------+-----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +======+========+========+=====+======+=====+=====+======+=====+ | SET1 | SID | ID1 | ID2 | ID3 | ID4 | ID5 | ID6 | ID7 | +------+--------+--------+-----+------+-----+-----+------+-----+ | | ID8 | etc. | | | | | | | +------+--------+--------+-----+------+-----+-----+------+-----+ | SET1 | 3 | 31 | 62 | 93 | 124 | 16 | 17 | 18 | +------+--------+--------+-----+------+-----+-----+------+-----+ | | 19 | | | | | | | | +------+--------+--------+-----+------+-----+-----+------+-----+ | SET1 | 6 | 29 | 32 | THRU | 50 | 61 | THRU | 70 | +------+--------+--------+-----+------+-----+-----+------+-----+ | | 17 | 57 | | | | | | | +------+--------+--------+-----+------+-----+-----+------+-----+ """ type = 'SET1'
[docs] @classmethod def _init_from_empty(cls): sid = 1 ids = [1] return SET1(sid, ids, is_skin=False, comment='')
def __init__(self, sid, ids, is_skin=False, comment=''): """ Creates a SET1 card, which defines a list of structural grid points or element identification numbers. Parameters ---------- sid : int set id ids : list[int, str] AECOMP, SPLINEx, PANEL : all grid points must exist XYOUTPUT : missing grid points are ignored The only valid string is THRU ``ids = [1, 3, 5, THRU, 10]`` is_skin : bool; default=False if is_skin is used; ids must be empty comment : str; default='' a comment for the card """ Set.__init__(self) if comment: self.comment = comment #: Unique identification number. (Integer > 0) self.sid = sid #: list of structural grid point or element identification numbers. #: (Integer > 0 or 'THRU'; for the 'THRU' option, ID1 < ID2 or 'SKIN'; #: in field 3) self.ids = expand_thru(ids, set_fields=False, sort_fields=False) #self.clean_ids() self.is_skin = is_skin self.xref_type = None self.ids_ref = None
[docs] @classmethod def add_card(cls, card, comment=''): """ Adds a SET1 card from ``BDF.add_card(...)`` Parameters ---------- card : BDFCard() a BDFCard object comment : str; default='' a comment for the card """ sid = integer(card, 1, 'sid') ids = fields(integer_or_string, card, 'ID', i=2, j=len(card)) is_skin = False i = 0 if len(ids) > 0: if isinstance(ids[0], str) and ids[0] == 'SKIN': is_skin = True i += 1 else: assert len(card) > 2, card return SET1(sid, ids[i:], is_skin=is_skin, comment=comment)
#def __eq__(self, set1): #assert self.type == set1.type, 'type=%s set1.type=%s' % (self.type, set1.type) #self.clean_ids() #set1.clean_ids() #if self.get_IDs() == set1.get_IDs(): #return True #return False
[docs] def symmetric_difference(self, set1): ids1 = set(self.get_ids()) ids2 = set(set1.get_ids()) return ids1.symmetric_difference(ids2)
[docs] def add_set(self, set1): self.ids += set1.get_ids() self.clean_ids()
[docs] def raw_fields(self): skin = [] if self.is_skin: skin = ['SKIN'] return ['SET1', self.sid] + skin + self.get_ids()
[docs] def cross_reference_set(self, model, xref_type, msg='', allow_empty_nodes=False): """ Cross links the card so referenced cards can be extracted directly Parameters ---------- model : BDF() the BDF object xref_type : str {'Node', 'Point'} allow_empty_nodes : bool; default=False do all nodes need to exist? SPLINEx, ACMODL, PANEL, AECOMP, XYOUTPUT - nodes - SPLINEx (all nodes must exist) - PANEL (all nodes must exist) - XYOUTPUT (missing nodes ignored) - AECOMP - ACMODL (optional) - elements - ACMODL (optional) """ msg = ', which is required by SET1 sid=%s%s' % (self.sid, msg) if xref_type == 'Node': self.ids_ref = model.Nodes(self.get_ids(), msg=msg) elif xref_type == 'Point': self.ids_ref = model.Points(self.get_ids(), msg=msg) else: raise NotImplementedError("xref_type=%r and must be ['Node', 'Point']" % xref_type) self.xref_type = xref_type
[docs] def safe_cross_reference(self, model: BDF, xref_type, msg='', allow_empty_nodes=False): """ Cross links the card so referenced cards can be extracted directly Parameters ---------- model : BDF() the BDF object xref_type : str {'Node'} allow_empty_nodes : bool; default=False do all nodes need to exist? SPLINEx, ACMODL, PANEL, AECOMP, XYOUTPUT - nodes - SPLINEx (all nodes must exist) - PANEL (all nodes must exist) - XYOUTPUT (missing nodes ignored) - AECOMP - ACMODL (optional) - elements - ACMODL (optional) """ assert msg != '' msg = ', which is required by SET1 sid=%s%s' % (self.sid, msg) if xref_type == 'Node': self.ids_ref, out = model.safe_get_nodes(self.get_ids(), msg=msg) if len(out): model.log.warning(out) elif xref_type == 'Point': self.ids_ref, out = model.safe_points(self.get_ids(), msg=msg) else: raise NotImplementedError("xref_type=%r and must be ['Node', 'Point']" % xref_type) self.xref_type = xref_type
[docs] def uncross_reference(self) -> None: """Removes cross-reference links""" if self.xref_type in ['Node', 'Point']: self.ids = self.get_ids() self.xref_type = None else: raise NotImplementedError("xref_type=%r and must be ['Node']" % self.xref_type) self.ids_ref = None
[docs] def get_ids(self): if self.ids_ref is None: return self.ids if self.xref_type is None: ids = self.ids elif self.xref_type in ['Node', 'Point']: ids = [node if isinstance(node, integer_types) else node.nid for node in self.ids_ref] else: raise NotImplementedError("xref_type=%r and must be ['Node']" % self.xref_type) return ids
[docs] def write_card(self, size: int=8, is_double: bool=False) -> str: skin = [] if self.is_skin: skin = ['SKIN'] # checked in NX 2014 / MSC 2005.1 card = ['SET1', self.sid] + skin + self.get_ids() return write_card(self.comment, card, size, is_double)
# I thought this worked in the new MSC Nastran... # Doesn't work in NX 2014 / MSC 2005.1 (multiple duplicate sids). # It may work with one sid, with singles and doubles on one card. #field_packs = [] #singles, doubles = collapse_thru_packs(self.get_ids()) #if singles: #field_packs.append(['SET1', self.sid] + skin + singles) #if doubles: #for pack in doubles: #field_packs.append(['SET1', self.sid] + skin + pack) #msg = [] #for field_pack in field_packs: #msg.append(print_card_8(field_pack)) #return ''.join(msg)
[docs] class SET2(Set): """ Defines a list of structural grid points in terms of aerodynamic macro elements. +------+--------+-------+-----+------+-----+-----+------+------+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +======+========+=======+=====+======+=====+=====+======+======+ | SET2 | SID | MACRO | SP1 | SP2 | CH1 | CH2 | ZMAX | ZMIN | +------+--------+-------+-----+------+-----+-----+------+------+ | SET2 | 3 | 111 | 0.0 | 0.75 | 0.0 |0.667| 3.51 | | +------+--------+-------+-----+------+-----+-----+------+------+ | SET2 | 6 | 222 | 0.0 | 0.75 | 0.0 |0.667| 3.51 | -1.0 | +------+--------+-------+-----+------+-----+-----+------+------+ SET2 entries are referenced by: - SPLINEi """ type = 'SET2'
[docs] @classmethod def _init_from_empty(cls): sid = 1 macro = 1 sp1 = 0. sp2 = 1. ch1 = 0. ch2 = 1. return SET2(sid, macro, sp1, sp2, ch1, ch2, comment='')
def __init__(self, sid: int, macro: int, sp1: float, sp2: float, ch1: float, ch2: float, zmax: float=0.0, zmin: float=0.0, comment: str='') -> None: """ Creates a SET2 card, which sefines a list of structural grid points in terms of aerodynamic macro elements. Remarks: - Points exactly on the boundary may be missed; therefore, to get all the grid points within the area of the macro element, SP1=-0.01, SP2=1.01, etc. should be used. - Use DIAG 18 to print the internal grid Ids found. Parameters ---------- sid : int set id macro : int the aerodynamic macro element id sp1 / sp2 : float lower/higher span division point defining the prism containing the set ch1 / ch2 : float lower/higher chord division point defining the prism containing the set zmax / zmin : float; default=0.0/0.0 z-coordinate of top/bottom of the prism containing the set a zero value implies a value of infinity """ Set.__init__(self) if comment: self.comment = comment #: Unique identification number. (Integer > 0) self.sid = sid #: Aerodynamic Macro Element ID. (Integer > 0) self.macro = macro #: Division Points spanwise and chordwise for the selection prism. (Real) self.sp1 = sp1 self.sp2 = sp2 self.ch1 = ch1 self.ch2 = ch2 #: Height limits for the selection prism. (Real) self.zmax = zmax self.zmin = zmin self.xref_type = None self.macro_ref = None
[docs] @classmethod def add_card(cls, card, comment=''): """ Adds a SET2 card from ``BDF.add_card(...)`` Parameters ---------- card : BDFCard() a BDFCard object comment : str; default='' a comment for the card """ sid = integer(card, 1, 'sid') macro = integer(card, 2, 'macro') sp1 = double(card, 3, 'sp1') sp2 = double(card, 4, 'sp2') ch1 = double(card, 5, 'ch1') ch2 = double(card, 6, 'ch2') zmax = double_or_blank(card, 7, 'zmax', 0.0) zmin = double_or_blank(card, 8, 'zmin', 0.0) return SET2(sid, macro, sp1, sp2, ch1, ch2, zmax=zmax, zmin=zmin, comment=comment)
[docs] def raw_fields(self): return ['SET2', self.sid, self.macro, self.sp1, self.sp2, self.ch1, self.ch2, self.zmax, self.zmin]
[docs] def cross_reference_set(self, model, xref_type: Optional[str], msg=''): """ Cross links the card so referenced cards can be extracted directly Parameters ---------- model : BDF() the BDF object xref_type : str {'MACRO'} i.e. the CAEROi elements """ msg = f', which is required by SET2 sid={self.sid}{msg}' if xref_type == 'MACRO': self.macro_ref = model.CAero(self.macro, msg=msg) else: raise NotImplementedError(f"xref_type={xref_type!r} and must be ['MACRO']") self.xref_type = xref_type
[docs] def get_ids(self): return []
[docs] def safe_cross_reference(self, model: BDF, xref_type: Optional[str], msg=''): msg = f', which is required by SET2 sid={self.sid}{msg}' if xref_type == 'MACRO': self.macro_ref = model.CAero(self.macro, msg=msg) else: model.log.error(f"xref_type={xref_type!r} and must be ['MACRO']") return self.xref_type = xref_type
#self.cross_reference_set(model, xref_errors, msg=msg)
[docs] def uncross_reference(self): if self.xref_type == 'MACRO': self.xref_type = None else: raise NotImplementedError(f"xref_type={self.xref_type!r} and must be ['MACRO']") self.macro_ref = None
[docs] class SET3(Set): """ Defines a list of grids, elements or points. SET3 entries are referenced by: - NX - ACMODL - PANEL - MSC - PBMSECT - PBRSECT - RFORCE - ELEM only (SOL 600) - DEACTEL - ELEM only (SOL 400) - RBAR, RBAR1, RBE1, RBE2, RBE2GS, RBE3, RROD, RSPLINE, RSSCON, RTRPLT and RTRPLT1 - RBEin / RBEex only - ELSIDi / XELSIDi - ELEM only - NDSIDi - GRID only +------+-----+-------+-----+-----+-----+-----+-----+-----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +======+=====+=======+=====+=====+=====+=====+=====+=====+ | SET3 | SID | DES | ID1 | ID2 | ID3 | ID4 | ID5 | ID6 | +------+-----+-------+-----+-----+-----+-----+-----+-----+ | | ID7 | ID8 | etc | | | | | | +------+-----+-------+-----+-----+-----+-----+-----+-----+ | SET3 | 1 | POINT | 11 | 12 | | | | | +------+-----+-------+-----+-----+-----+-----+-----+-----+ """ type = 'SET3' valid_descs = ['GRID', 'POINT', 'ELEMENT', 'PROP', 'RBEin', 'RBEex']
[docs] @classmethod def _init_from_empty(cls): sid = 1 desc = 'ELEM' ids = [1] return SET3(sid, desc, ids, comment='')
def __init__(self, sid: int, desc: str, ids: list[int], comment: str=''): Set.__init__(self) if comment: self.comment = comment #: Unique identification number. (Integer > 0) self.sid = sid #: Set description (Character). Valid options are 'GRID', 'ELEM', #: 'POINT' and 'PROP'. if desc == 'ELEM': desc = 'ELEMENT' elif desc == 'RBEIN': desc = 'RBEin' elif desc == 'RBEEX': desc = 'RBEex' self.desc = desc #: Identifiers of grids points, elements, points or properties. #: (Integer > 0) ids = [idi for idi in ids if idi is not None] self.ids = expand_thru(ids, set_fields=False, sort_fields=False) self.ids_ref = None self.xref_type = None
[docs] def validate(self): if self.desc not in self.valid_descs: msg = 'desc=%r; valid_descs=[%s]' % (self.desc, ', '.join(self.valid_descs)) raise ValueError(msg)
[docs] def get_ids(self): if self.ids_ref is None: return self.ids if self.xref_type is None: ids = self.ids elif self.xref_type == 'Point': # TODO: improve this... ids = [point if isinstance(point, integer_types) else point.nid for point in self.ids_ref] else: # 'Node', raise NotImplementedError("xref_type=%r and must be ['Point']" % self.xref_type) return ids
[docs] def cross_reference_set(self, model, xref_type, msg=''): msg = ', which is required by SET3 sid=%s%s' % (self.sid, msg) if xref_type == 'GRID': # was 'Node' # not tested relative to Nastran, seems obvious though # I'm not sure why Node was here vs. GRID # the block was disabled anyways, so probably doesn't matter self.ids = model.Nodes(self.get_ids(), msg=msg) if xref_type == 'Point': self.ids_ref = model.Points(self.get_ids(), msg=msg) else: raise NotImplementedError("xref_type=%r and must be ['Point']" % xref_type) self.xref_type = xref_type
[docs] def add_set(self, set3): self.ids += set3.get_ids() assert self.sid == set3.sid, 'SET3.sid=%r; existing sid=%r new=%r' % (self.sid, self.sid, set3.sid) assert self.desc == set3.desc, 'SET3.sid=%r; existing desc=%r new=%r' % (self.sid, self.desc, set3.desc) self.clean_ids()
[docs] @classmethod def add_card(cls, card, comment=''): """ Adds a SET3 card from ``BDF.add_card(...)`` Parameters ---------- card : BDFCard() a BDFCard object comment : str; default='' a comment for the card """ sid = integer(card, 1, 'sid') desc = string(card, 2, 'desc') ids = fields(integer_string_or_blank, card, 'ID', i=3, j=len(card)) return SET3(sid, desc, ids, comment=comment)
[docs] def union(self, set3): assert self.type == set3.type, 'type=%r set3.type=%r' % (self.type, set3.type) assert self.desc == set3.desc, 'self.desc=%r set3.desc=%r' % (self.desc, set3.desc) ids1 = set(self.ids) ids2 = set(set3.ids) self.ids = list(ids1.union(ids2))
[docs] def symmetric_difference(self, set3): assert self.type == set3.type, 'type=%r set3.type=%r' % (self.type, set3.type) ids1 = set(self.ids) ids2 = set(set3.ids) return ids1.symmetric_difference(ids2)
[docs] def is_grid(self): if self.desc == 'GRID': return True return False
[docs] def is_point(self): if self.desc == 'POINT': return True return False
[docs] def is_property(self): if self.desc == 'PROP': return True return False
[docs] def is_element(self): if self.desc == 'ELEMENT': return True return False
[docs] def SetIDs(self, collapse=True): """gets the IDs of the SETx""" if collapse: return collapse_thru(self.ids, nthru=1) else: return self.ids
[docs] def raw_fields(self): """Gets the "raw" card without any processing as a list for printing""" list_fields = ['SET3', self.sid, self.desc] + self.SetIDs() return list_fields
def __repr__(self): #fields_blocks = [ #'SET3', #[[self.sid, self.desc], False], # these are not all integers #[self.SetIDs(), True], # these are all integers #] #print(fields_blocks) #return self.comment + print_int_card_blocks(fields_blocks) msg = self.comment self.ids.sort() ids = self.get_ids() packs = condense(ids) if len(packs) == 1: singles, doubles = build_thru_packs(packs, max_dv=1) packs = collapse_thru(ids) for pack in doubles: msg += print_card_8(['SET3', self.sid, self.desc] + pack) if singles: msg += print_card_8(['SET3', self.sid, self.desc] + singles) else: msg += print_card_8(['SET3', self.sid, self.desc] + ids) return msg
[docs] def write_card(self, size: int=8, is_double: bool=False) -> str: return str(self)
[docs] class SESET(SetSuper): """ Defines interior grid points for a superelement. """ type = 'SESET'
[docs] @classmethod def _init_from_empty(cls): seid = 1 ids = [1, 2] return SESET(seid, ids, comment='')
def __init__(self, seid, ids, comment=''): SetSuper.__init__(self) if comment: self.comment = comment self.seid = seid #: Grid or scalar point identification number. #: (0 < Integer < 1000000; G1 < G2) self.ids = expand_thru(ids) self.clean_ids()
[docs] @classmethod def add_card(cls, card, comment=''): """ Adds a SESET card from ``BDF.add_card(...)`` Parameters ---------- card : BDFCard() a BDFCard object comment : str; default='' a comment for the card """ seid = integer_or_blank(card, 1, 'seid', 0) ids = fields(integer_or_string, card, 'ID', i=2, j=len(card)) return SESET(seid, ids, comment=comment)
[docs] def add_seset(self, seset): self.ids += seset.ids self.clean_ids()
[docs] def raw_fields(self): list_fields = ['SESET', self.seid] + collapse_thru(self.ids) return list_fields
def __repr__(self): thru_fields = collapse_thru(self.ids) #list_fields = ['SESET', self.seid] cards = [] while 'THRU' in thru_fields: ithru = thru_fields.index('THRU') card = print_card_8(['SESET', self.seid] + thru_fields[ithru - 1:ithru + 2]) cards.append(card) thru_fields = thru_fields[0:ithru - 1]+thru_fields[ithru + 2:] if thru_fields: card = print_card_8(['SESET', self.seid] + thru_fields) cards.append(card) return ''.join(cards)
[docs] def cross_reference(self, model: BDF) -> None: pass
[docs] def uncross_reference(self) -> None: """Removes cross-reference links""" pass
[docs] class SEBSET(SuperABCQSet): """ Defines boundary degrees-of-freedom to be fixed (b-set) during generalized dynamic reduction or component mode calculations. +--------+------+-----+------+-----+----+-----+----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | +========+======+=====+======+=====+====+=====+====+ | SEBSET | SEID | ID1 | C1 | ID2 | C2 | ID3 | C3 | +--------+------+-----+------+-----+----+-----+----+ | SEBSET | C | ID1 | THRU | ID2 | | | | +--------+------+-----+------+-----+----+-----+----+ """ type = 'SEBSET' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): seid = 1 ids = [1, 2] components = ['123', '456'] return SEBSET(seid, ids, components, comment='')
def __init__(self, seid, ids, components, comment=''): SuperABCQSet.__init__(self, seid, ids, components, comment)
[docs] class SEBSET1(SuperABCQSet1): """ Defines boundary degrees-of-freedom to be fixed (b-set) during generalized dynamic reduction or component mode synthesis calculations. +----------+------+-----+------+------+-----+-----+-----+-----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +==========+======+=====+======+======+=====+=====+=====+=====+ | SEBSET1 | SEID | C | ID1 | ID2 | ID3 | ID4 | ID5 | ID6 | +----------+------+-----+------+------+-----+-----+-----+-----+ | | ID7 | ID9 | | | | | | | +----------+------+-----+------+------+-----+-----+-----+-----+ | SEBSET1 | SEID | C | ID1 | THRU | ID2 | | | | +----------+------+-----+------+------+-----+-----+-----+-----+ """ type = 'SEBSET1' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): seid = 1 ids = [1, 2] components = '123' return SEBSET1(seid, ids, components, comment='')
def __init__(self, seid, ids, components, comment=''): SuperABCQSet1.__init__(self, seid, ids, components, comment)
[docs] class SECSET(SuperABCQSet): type = 'SECSET' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): seid = 1 ids = [1, 2] components = ['123', '456'] return SECSET(seid, ids, components, comment='')
def __init__(self, seid, ids, components, comment=''): SuperABCQSet.__init__(self, seid, ids, components, comment)
[docs] class SECSET1(SuperABCQSet1): """ Defines SECSET1 +----------+------+-----+------+------+-----+-----+-----+-----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +==========+======+=====+======+======+=====+=====+=====+=====+ | SECSET1 | SEID | C | ID1 | ID2 | ID3 | ID4 | ID5 | ID6 | +----------+------+-----+------+------+-----+-----+-----+-----+ | | ID7 | ID9 | | | | | | | +----------+------+-----+------+------+-----+-----+-----+-----+ | SECSET1 | SEID | C | ID1 | THRU | ID2 | | | | +----------+------+-----+------+------+-----+-----+-----+-----+ """ type = 'SECSET1' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): seid = 1 ids = [1, 2] components = '123' return SECSET1(seid, ids, components, comment='')
def __init__(self, seid, ids, components, comment=''): SuperABCQSet1.__init__(self, seid, ids, components, comment)
[docs] class SEQSET(SuperABCQSet): type = 'SEQSET' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): seid = 1 ids = [1, 2] components = ['123', '456'] return SEQSET(seid, ids, components, comment='')
def __init__(self, seid, ids, components, comment=''): SuperABCQSet.__init__(self, seid, ids, components, comment)
[docs] class SEQSET1(SuperABCQSet1): type = 'SEQSET1' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): seid = 1 ids = [1, 2] components = '123' return SEQSET1(seid, ids, components, comment='')
def __init__(self, seid, ids, components, comment=''): SuperABCQSet1.__init__(self, seid, ids, components, comment)
[docs] class SEQSEP(SetSuper): # not integrated...is this an SESET ??? """ Used with the CSUPER entry to define the correspondence of the exterior grid points between an identical or mirror-image superelement and its primary superelement. """ type = 'SEQSEP' def __init__(self, ssid, psid, ids, comment=''): SetSuper.__init__(self) if comment: self.comment = comment #: Identification number for secondary superelement. (Integer >= 0). self.ssid = ssid #: Identification number for the primary superelement. (Integer >= 0). self.psid = psid #: Exterior grid point identification numbers for the primary #: superelement. (Integer > 0) self.ids = expand_thru(ids) self.clean_ids()
[docs] @classmethod def add_card(cls, card, comment=''): """ Adds a SEQSEP card from ``BDF.add_card(...)`` Parameters ---------- card : BDFCard() a BDFCard object comment : str; default='' a comment for the card """ ssid = integer(card, 1, 'ssid') psid = integer(card, 2, 'psid') ids = fields(integer_or_string, card, 'ID', i=3, j=len(card)) return SEQSEP(ssid, psid, ids, comment=comment)
[docs] def get_ids(self)-> list[int]: """gets the ids""" return self.ids
[docs] def raw_fields(self): """gets the "raw" card without any processing as a list for printing""" list_fields = ['SEQSEP', self.ssid, self.psid] + self.get_ids() return list_fields
[docs] class RADSET(ABCQSet1): """ Specifies which radiation cavities are to be included for radiation enclosure analysis. +--------+----------+----------+----------+----------+----------+----------+----------+----------+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +========+==========+==========+==========+==========+==========+==========+==========+==========+ | RADSET | ICAVITY1 | ICAVITY2 | ICAVITY3 | ICAVITY4 | ICAVITY5 | ICAVITY6 | ICAVITY7 | ICAVITY8 | +--------+----------+----------+----------+----------+----------+----------+----------+----------+ | | ICAVITY9 | | | | | | | | +--------+----------+----------+----------+----------+----------+----------+----------+----------+ | RADSET | 1 | 2 | 3 | 4 | | | | | +--------+----------+----------+----------+----------+----------+----------+----------+----------+ """ type = 'RADSET'
[docs] @classmethod def _init_from_empty(cls): cavities = [1, 2] return RADSET(cavities, comment='')
[docs] def _finalize_hdf5(self, encoding): """hdf5 helper function""" if isinstance(self.cavities, np.ndarray): self.cavities = self.cavities.tolist()
def __init__(self, cavities, comment=''): """ Creates a RADSET card Parameters ---------- cavities : list[int] the RADCAV ids comment : str; default='' a comment for the card """ if comment: self.comment = comment self.cavities = cavities #: Identifiers of grids points. (Integer > 0) #self.ids = expand_thru(ids) #self.ids_ref = None
[docs] @classmethod def add_card(cls, card: BDFCard, comment: str=''): """ Adds a RADSET card from ``BDF.add_card(...)`` Parameters ---------- card : BDFCard() a BDFCard object comment : str; default='' a comment for the card """ nfields = len(card) cavities = [] i = 1 for ifield in range(1, nfields): cavity = integer(card, ifield, 'iCavity%i' % i) if cavity: i += 1 cavities.append(cavity) return RADSET(cavities, comment=comment)
#def cross_reference(self, model: BDF) -> None: #""" #Cross links the card so referenced cards can be extracted directly #Parameters #---------- #model : BDF() #the BDF object #""" #msg = ', which is required by USET1 name=%s' % (self.name) #self.ids_ref = model.EmptyNodes(self.node_ids, msg=msg) #def uncross_reference(self) -> None: #self.ids = self.node_ids #self.ids_ref = None
[docs] def raw_fields(self): """gets the "raw" card without any processing as a list for printing""" list_fields = ['RADSET'] + self.cavities # collapse_thru(self.node_ids) return list_fields
def __repr__(self): list_fields = self.raw_fields() return self.comment + print_card_8(list_fields)
[docs] class USET(Set): """ Defines a degrees-of-freedom set. +------+-------+-----+------+-----+----+-----+----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | +======+=======+=====+======+=====+====+=====+====+ | USET | SNAME | ID1 | C1 | ID2 | C2 | ID3 | C3 | +------+-------+-----+------+-----+----+-----+----+ | USET | JUNK | ID1 | THRU | ID2 | | | | +------+-------+-----+------+-----+----+-----+----+ """ type = 'USET' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): #name = 'SNAME' ids = [1, 2] components = ['123', '456'] return QSET(ids, components, comment='')
def __init__(self, name, ids, components, comment=''): """ Creates a USET card, which defines a degrees-of-freedom set. Parameters ---------- name : str SNAME Set name. (One to four characters or the word 'ZERO' followed by the set name.) ids : list[int] the GRID/SPOINT ids components : list[str] the degree of freedoms (e.g., '1', '123') comment : str; default='' a comment for the card """ Set.__init__(self) if comment: self.comment = comment self.name = name #: Identifiers of grids points. (Integer > 0) self.components = components self.ids = ids self.ids_ref = None
[docs] @classmethod def add_card(cls, card, comment=''): """ Adds a USET card from ``BDF.add_card(...)`` Parameters ---------- card : BDFCard() a BDFCard object comment : str; default='' a comment for the card """ name = string(card, 1, 'name') components = [] ids = [] nsets = (len(card) - 1) // 2 for iset in range(nsets): i = iset * 2 + 2 idi = integer(card, i, 'node_id' + str(iset)) component = parse_components(card, i + 1, 'component' + str(iset)) components.append(component) ids.append(idi) return USET(name, ids, components, comment=comment)
@classmethod def add_op2_data(cls, data, comment=''): """ tested by gspc1.op2 for some reason, the setname is an integer and has bizarre rules that I don't understand like: - the setname is 1-4 characters, except if it's 'ZERO%i' % sid ummm...odd """ sid = data[0] nid = data[1] if sid < 0: name = 'ZERO' else: comment = 'sid=%s (???)' % sid name = 'U%i' % nid assert nid > 0, nid component = str(data[2]) for componenti in component: assert componenti in '0123456', component return USET(name, [nid], [component], comment=comment)
[docs] def cross_reference(self, model: BDF) -> None: """ Cross links the card so referenced cards can be extracted directly Parameters ---------- model : BDF() the BDF object """ msg = ', which is required by USET name=%s' % (self.name) self.ids_ref = model.EmptyNodes(self.node_ids, msg=msg)
[docs] def uncross_reference(self) -> None: """Removes cross-reference links""" self.ids = self.node_ids self.ids_ref = None
@property def node_ids(self): if self.ids_ref is None: return self.ids msg = ', which is required by USET name=%s' % (self.name) return _node_ids(self, self.ids_ref, allow_empty_nodes=True, msg=msg)
[docs] def raw_fields(self): """ gets the "raw" card without any processing as a list for printing """ list_fields = ['USET', self.name] for (component, idi) in zip(self.components, self.node_ids): list_fields += [idi, component] return list_fields
[docs] class USET1(ABCQSet1): """ Defines a degree-of-freedom set. +-------+-------+-----+------+------+-----+-----+-----+-----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | +=======+=======+=====+======+======+=====+=====+=====+=====+ | USET1 | SNAME | C | ID2 | ID3 | ID4 | ID5 | ID6 | ID7 | +-------+-------+-----+------+------+-----+-----+-----+-----+ | | ID9 | | | | | | | | +-------+-------+-----+------+------+-----+-----+-----+-----+ | USET1 | SNAME | C | ID1 | THRU | ID2 | | | | +-------+-------+-----+------+------+-----+-----+-----+-----+ """ type = 'USET1' _properties = ['node_ids']
[docs] @classmethod def _init_from_empty(cls): name = 'SNAME' ids = [1, 2] components = '123' return USET1(name, ids, components, comment='')
def __init__(self, name, ids, components, comment=''): """ Creates a USET1 card, which defines a degrees-of-freedom set. Parameters ---------- name : str SNAME Set name. (One to four characters or the word 'ZERO' followed by the set name.) ids : list[int] the GRID/SPOINT ids components : str the degree of freedoms (e.g., '1', '123') comment : str; default='' a comment for the card """ ABCQSet1.__init__(self, ids, components, comment=comment) #if comment: #self.comment = comment self.name = name #: Component number. (Integer zero or blank for scalar points or any #: unique combination of the Integers 1 through 6 for grid points with #: no embedded blanks.) #self.components = components #: Identifiers of grids points. (Integer > 0) #self.ids = expand_thru(ids) #self.ids_ref = None
[docs] @classmethod def add_card(cls, card, comment=''): """ Adds a USET1 card from ``BDF.add_card(...)`` Parameters ---------- card : BDFCard() a BDFCard object comment : str; default='' a comment for the card """ name = string(card, 1, 'name') components = fcomponents_or_blank(card, 2, 'components', '0') nfields = len(card) ids = [] i = 1 for ifield in range(3, nfields): idi = integer_string_or_blank(card, ifield, 'ID%i' % i) if idi: i += 1 ids.append(idi) return USET1(name, ids, components, comment=comment)
@classmethod def add_op2_data(cls, data, comment=''): """ tested by gspc1.op2 for some reason, the setname is an integer and has bizarre rules that I don't understand like: - the setname is 1-4 characters, except if it's 'ZERO%i' % sid ummm...odd """ name, components, ids = data #sid = data[0] #nid = data[1] #if sid < 0: #name = 'ZERO' #else: #comment = 'sid=%s (???)' % sid #name = 'U%i' % nid #assert nid > 0, nid #component = str(data[2]) for component in components: assert component in '0123456', components return USET1(name, ids, components, comment=comment)
[docs] def cross_reference(self, model: BDF) -> None: """ Cross links the card so referenced cards can be extracted directly Parameters ---------- model : BDF() the BDF object """ msg = ', which is required by USET1 name=%s' % (self.name) self.ids_ref = model.EmptyNodes(self.node_ids, msg=msg)
[docs] def uncross_reference(self) -> None: """Removes cross-reference links""" self.ids = self.node_ids self.ids_ref = None
@property def node_ids(self): if self.ids_ref is None: return self.ids msg = ', which is required by USET1 name=%s' % (self.name) return _node_ids(self, self.ids_ref, allow_empty_nodes=True, msg=msg)
[docs] def raw_fields(self): """gets the "raw" card without any processing as a list for printing""" list_fields = ['USET1', self.name, self.components] + collapse_thru(self.node_ids) return list_fields
def __repr__(self): list_fields = self.raw_fields() return self.comment + print_card_8(list_fields)