Make records store their associated Property records.

Previously, `Property` records were (incorrectly) just associated with
the library or cell.

This requires a change to how `CellName`s are handled by `OasisLayout`,
since they can have associated properties. They now have their own
non-record object (like `XName`s did before) which holds the properties.

There is also now a `FileModals.property_target` attribute which tracks
which record new `Property` rescords should be associated with.
This commit is contained in:
Jan Petykiewicz 2020-09-10 20:03:01 -07:00
parent 167b16e1c9
commit 3627b63658
2 changed files with 137 additions and 36 deletions

View File

@ -30,11 +30,16 @@ class FileModals:
textstring_implicit: Optional[bool] = None textstring_implicit: Optional[bool] = None
propstring_implicit: Optional[bool] = None propstring_implicit: Optional[bool] = None
property_target: List[records.Property]
within_cell: bool = False within_cell: bool = False
within_cblock: bool = False within_cblock: bool = False
end_has_offset_table: bool end_has_offset_table: bool = False
started: bool = False started: bool = False
def __init__(self, property_target = List[records.Property]):
self.property_target = property_target
class OasisLayout: class OasisLayout:
""" """
@ -53,7 +58,7 @@ class OasisLayout:
validation (Validation): checksum data validation (Validation): checksum data
(Names) (Names)
cellnames (Dict[int, NString]): Cell names cellnames (Dict[int, CellName]): Cell names
propnames (Dict[int, NString]): Property names propnames (Dict[int, NString]): Property names
xnames (Dict[int, XName]): Custom names xnames (Dict[int, XName]): Custom names
@ -73,7 +78,7 @@ class OasisLayout:
properties: List[records.Property] properties: List[records.Property]
cells: List['Cell'] cells: List['Cell']
cellnames: Dict[int, NString] cellnames: Dict[int, 'CellName']
propnames: Dict[int, NString] propnames: Dict[int, NString]
xnames: Dict[int, 'XName'] xnames: Dict[int, 'XName']
@ -115,9 +120,9 @@ class OasisLayout:
Returns: Returns:
New `OasisLayout` object. New `OasisLayout` object.
""" """
file_state = FileModals()
modals = Modals()
layout = OasisLayout(unit=-1) # dummy unit layout = OasisLayout(unit=-1) # dummy unit
modals = Modals()
file_state = FileModals(layout.properties)
read_magic_bytes(stream) read_magic_bytes(stream)
@ -225,7 +230,10 @@ class OasisLayout:
key = record.reference_number key = record.reference_number
if key is None: if key is None:
key = len(self.cellnames) key = len(self.cellnames)
self.cellnames[key] = record.nstring
cellname = CellName.from_record(record)
self.cellnames[key] = cellname
file_state.property_target = cellname.properties
elif record_id in (5, 6): elif record_id in (5, 6):
implicit = record_id == 5 implicit = record_id == 5
if file_state.textstring_implicit is None: if file_state.textstring_implicit is None:
@ -272,10 +280,7 @@ class OasisLayout:
elif record_id in (28, 29): elif record_id in (28, 29):
record = records.Property.read(stream, record_id) record = records.Property.read(stream, record_id)
record.merge_with_modals(modals) record.merge_with_modals(modals)
if not file_state.within_cell: file_state.property_target.append(record)
self.properties.append(record)
else:
self.cells[-1].properties.append(record)
elif record_id in (30, 31): elif record_id in (30, 31):
implicit = record_id == 30 implicit = record_id == 30
if file_state.xname_implicit is None: if file_state.xname_implicit is None:
@ -289,6 +294,7 @@ class OasisLayout:
if key is None: if key is None:
key = len(self.xnames) key = len(self.xnames)
self.xnames[key] = XName.from_record(record) self.xnames[key] = XName.from_record(record)
# TODO: do anything with property target?
# #
# Cell and elements # Cell and elements
@ -296,7 +302,9 @@ class OasisLayout:
elif record_id in (13, 14): elif record_id in (13, 14):
record = records.Cell.read(stream, record_id) record = records.Cell.read(stream, record_id)
record.merge_with_modals(modals) record.merge_with_modals(modals)
self.cells.append(Cell(record.name)) cell = Cell(record.name)
self.cells.append(cell)
file_state.property_target = cell.properties
elif record_id in (15, 16): elif record_id in (15, 16):
record = records.XYMode.read(stream, record_id) record = records.XYMode.read(stream, record_id)
record.merge_with_modals(modals) record.merge_with_modals(modals)
@ -304,10 +312,12 @@ class OasisLayout:
record = records.Placement.read(stream, record_id) record = records.Placement.read(stream, record_id)
record.merge_with_modals(modals) record.merge_with_modals(modals)
self.cells[-1].placements.append(record) self.cells[-1].placements.append(record)
file_state.property_target = record.properties
elif record_id in _GEOMETRY: elif record_id in _GEOMETRY:
record = _GEOMETRY[record_id].read(stream, record_id) record = _GEOMETRY[record_id].read(stream, record_id)
record.merge_with_modals(modals) record.merge_with_modals(modals)
self.cells[-1].geometry.append(record) self.cells[-1].geometry.append(record)
file_state.property_target = record.properties
else: else:
raise InvalidRecordError('Unknown record id: {}'.format(record_id)) raise InvalidRecordError('Unknown record id: {}'.format(record_id))
return False return False
@ -330,10 +340,12 @@ class OasisLayout:
size = 0 size = 0
size += write_magic_bytes(stream) size += write_magic_bytes(stream)
size += records.Start(self.unit, self.version).dedup_write(stream, modals) size += records.Start(self.unit, self.version).dedup_write(stream, modals)
size += sum(p.dedup_write(stream, modals) for p in self.properties)
cellnames_offset = OffsetEntry(False, size) cellnames_offset = OffsetEntry(False, size)
size += sum(records.CellName(name, refnum).dedup_write(stream, modals) for refnum, cn in self.cellnames.items():
for refnum, name in self.cellnames.items()) size += records.CellName(cn.nstring, refnum).dedup_write(stream, modals)
size += sum(p.dedup_write(stream, modals) for p in cn.properties)
propnames_offset = OffsetEntry(False, size) propnames_offset = OffsetEntry(False, size)
size += sum(records.PropName(name, refnum).dedup_write(stream, modals) size += sum(records.PropName(name, refnum).dedup_write(stream, modals)
@ -354,8 +366,6 @@ class OasisLayout:
layernames_offset = OffsetEntry(False, size) layernames_offset = OffsetEntry(False, size)
size += sum(r.dedup_write(stream, modals) for r in self.layers) size += sum(r.dedup_write(stream, modals) for r in self.layers)
size += sum(p.dedup_write(stream, modals) for p in self.properties)
size += sum(c.dedup_write(stream, modals) for c in self.cells) size += sum(c.dedup_write(stream, modals) for c in self.cells)
offset_table = OffsetTable( offset_table = OffsetTable(
@ -386,15 +396,17 @@ class Cell:
placements: List[records.Placement] placements: List[records.Placement]
geometry: List[records.geometry_t] geometry: List[records.geometry_t]
def __init__(self, name: Union[NString, str, int]): def __init__(self,
""" name: Union[NString, str, int],
Args: *,
name: `NString` or "CellName reference" number properties: Optional[List[records.Property]] = None,
""" placements: Optional[List[records.Placement]] = None,
geometry: Optional[List[records.geometry_t]] = None,
):
self.name = name if isinstance(name, (NString, int)) else NString(name) self.name = name if isinstance(name, (NString, int)) else NString(name)
self.properties = [] self.properties = [] if properties is None else properties
self.placements = [] self.placements = [] if placements is None else placements
self.geometry = [] self.geometry = [] if geometry is None else geometry
def dedup_write(self, stream: io.BufferedIOBase, modals: Modals) -> int: def dedup_write(self, stream: io.BufferedIOBase, modals: Modals) -> int:
""" """
@ -413,11 +425,54 @@ class Cell:
""" """
size = records.Cell(self.name).dedup_write(stream, modals) size = records.Cell(self.name).dedup_write(stream, modals)
size += sum(p.dedup_write(stream, modals) for p in self.properties) size += sum(p.dedup_write(stream, modals) for p in self.properties)
size += sum(p.dedup_write(stream, modals) for p in self.placements) for placement in self.placements:
size += sum(g.dedup_write(stream, modals) for g in self.geometry) size += placement.dedup_write(stream, modals)
size += sum(p.dedup_write(stream, modals) for p in placement.properties)
for shape in self.geometry:
size += shape.dedup_write(stream, modals)
size += sum(p.dedup_write(stream, modals) for p in shape.properties)
return size return size
class CellName:
"""
Representation of a CellName.
This class is effectively a simplified form of a `records.CellName`,
with the reference data stripped out.
"""
nstring: NString
properties: List[records.Property]
def __init__(self,
nstring: Union[NString, str],
properties: Optional[List[records.Property]] = None):
"""
Args:
nstring: The contained string.
properties: Properties which apply to this CellName's cell, but
are placed following the CellName record.
"""
if isinstance(nstring, NString):
self.nstring = nstring
else:
self.nstring = NString(nstring)
self.properties = [] if properties is None else properties
@staticmethod
def from_record(record: records.CellName) -> 'CellName':
"""
Create an `CellName` object from a `records.CellName` record.
Args:
record: CellName record to use.
Returns:
A new `CellName` object.
"""
return CellName(record.nstring)
class XName: class XName:
""" """
Representation of an XName. Representation of an XName.
@ -446,7 +501,7 @@ class XName:
record: XName record to use. record: XName record to use.
Returns: Returns:
`XName` object. a new `XName` object.
""" """
return XName(record.attribute, record.bstring) return XName(record.attribute, record.bstring)

View File

@ -1101,15 +1101,21 @@ class XElement(Record):
""" """
attribute: int attribute: int
bstring: bytes bstring: bytes
properties: List['Property']
def __init__(self, attribute: int, bstring: bytes): def __init__(self,
attribute: int,
bstring: bytes,
properties: Optional[List['Property']] = None):
""" """
Args: Args:
attribute: Attribute number. attribute: Attribute number.
bstring: Binary data for this XElement. bstring: Binary data for this XElement.
properties: List of property records associated with this record.
""" """
self.attribute = attribute self.attribute = attribute
self.bstring = bstring self.bstring = bstring
self.properties = [] if properties is None else properties
def merge_with_modals(self, modals: Modals): def merge_with_modals(self, modals: Modals):
pass pass
@ -1147,6 +1153,7 @@ class XGeometry(Record, GeometryMixin):
x (Optional[int]): None means reuse modal x (Optional[int]): None means reuse modal
y (Optional[int]): None means reuse modal y (Optional[int]): None means reuse modal
repetition (Optional[repetition_t]): Repetition, if any repetition (Optional[repetition_t]): Repetition, if any
properties (List[Property]): List of property records associate with this record.
""" """
attribute: int attribute: int
bstring: bytes bstring: bytes
@ -1155,6 +1162,7 @@ class XGeometry(Record, GeometryMixin):
x: Optional[int] = None x: Optional[int] = None
y: Optional[int] = None y: Optional[int] = None
repetition: Optional[repetition_t] = None repetition: Optional[repetition_t] = None
properties: List['Property']
def __init__(self, def __init__(self,
attribute: int, attribute: int,
@ -1163,7 +1171,8 @@ class XGeometry(Record, GeometryMixin):
datatype: Optional[int] = None, datatype: Optional[int] = None,
x: Optional[int] = None, x: Optional[int] = None,
y: Optional[int] = None, y: Optional[int] = None,
repetition: Optional[repetition_t] = None): repetition: Optional[repetition_t] = None,
properties: Optional[List['Property']] = None):
""" """
Args: Args:
attribute: Attribute number for this XGeometry. attribute: Attribute number for this XGeometry.
@ -1173,6 +1182,7 @@ class XGeometry(Record, GeometryMixin):
x: X-offset. Default `None` (use modal). x: X-offset. Default `None` (use modal).
y: Y-offset. Default `None` (use modal). y: Y-offset. Default `None` (use modal).
repetition: Repetition. Default `None` (no repetition). repetition: Repetition. Default `None` (no repetition).
properties: List of property records associated with this record.
""" """
self.attribute = attribute self.attribute = attribute
self.bstring = bstring self.bstring = bstring
@ -1181,6 +1191,7 @@ class XGeometry(Record, GeometryMixin):
self.x = x self.x = x
self.y = y self.y = y
self.repetition = repetition self.repetition = repetition
self.properties = [] if properties is None else properties
def merge_with_modals(self, modals: Modals): def merge_with_modals(self, modals: Modals):
adjust_coordinates(self, modals, 'geometry_x', 'geometry_y') adjust_coordinates(self, modals, 'geometry_x', 'geometry_y')
@ -1305,6 +1316,7 @@ class Placement(Record):
y (Optional[int]): y-offset, None means reuse modal y (Optional[int]): y-offset, None means reuse modal
repetition (repetition_t or None): Repetition, if any repetition (repetition_t or None): Repetition, if any
flip (bool): Whether to perform reflection about the x-axis. flip (bool): Whether to perform reflection about the x-axis.
properties (List[Property]): List of property records associate with this record.
""" """
name: Union[NString, int, None] = None name: Union[NString, int, None] = None
magnification: Optional[real_t] = None magnification: Optional[real_t] = None
@ -1313,6 +1325,7 @@ class Placement(Record):
y: Optional[int] = None y: Optional[int] = None
repetition: Optional[repetition_t] = None repetition: Optional[repetition_t] = None
flip: bool flip: bool
properties: List['Property']
def __init__(self, def __init__(self,
flip: bool, flip: bool,
@ -1321,7 +1334,8 @@ class Placement(Record):
angle: Optional[real_t] = None, angle: Optional[real_t] = None,
x: Optional[int] = None, x: Optional[int] = None,
y: Optional[int] = None, y: Optional[int] = None,
repetition: Optional[repetition_t] = None): repetition: Optional[repetition_t] = None,
properties: Optional[List['Property']] = None):
""" """
Args: Args:
flip: Whether to perform reflection about the x-axis. flip: Whether to perform reflection about the x-axis.
@ -1333,6 +1347,7 @@ class Placement(Record):
x: X-offset. Default `None` (use modal). x: X-offset. Default `None` (use modal).
y: Y-offset. Default `None` (use modal). y: Y-offset. Default `None` (use modal).
repetition: Repetition. Default `None` (no repetition). repetition: Repetition. Default `None` (no repetition).
properties: List of property records associated with this record.
""" """
self.x = x self.x = x
self.y = y self.y = y
@ -1344,6 +1359,7 @@ class Placement(Record):
self.name = name self.name = name
else: else:
self.name = NString(name) self.name = NString(name)
self.properties = [] if properties is None else properties
def get_name(self) -> Union[NString, int]: def get_name(self) -> Union[NString, int]:
return verify_modal(self.name) # type: ignore return verify_modal(self.name) # type: ignore
@ -1448,6 +1464,7 @@ class Text(Record, GeometryMixin):
x (Optional[int]): x-offset, None means reuse modal x (Optional[int]): x-offset, None means reuse modal
y (Optional[int]): y-offset, None means reuse modal y (Optional[int]): y-offset, None means reuse modal
repetition (Optional[repetition_t]): Repetition, if any repetition (Optional[repetition_t]): Repetition, if any
properties (List[Property]): List of property records associate with this record.
""" """
string: Optional[Union[AString, int]] = None string: Optional[Union[AString, int]] = None
layer: Optional[int] = None layer: Optional[int] = None
@ -1455,6 +1472,7 @@ class Text(Record, GeometryMixin):
x: Optional[int] = None x: Optional[int] = None
y: Optional[int] = None y: Optional[int] = None
repetition: Optional[repetition_t] = None repetition: Optional[repetition_t] = None
properties: List['Property']
def __init__(self, def __init__(self,
string: Union[AString, str, int, None] = None, string: Union[AString, str, int, None] = None,
@ -1462,7 +1480,8 @@ class Text(Record, GeometryMixin):
datatype: Optional[int] = None, datatype: Optional[int] = None,
x: Optional[int] = None, x: Optional[int] = None,
y: Optional[int] = None, y: Optional[int] = None,
repetition: Optional[repetition_t] = None): repetition: Optional[repetition_t] = None,
properties: Optional[List['Property']] = None):
""" """
Args: Args:
string: Text content, or `TextString` reference number. string: Text content, or `TextString` reference number.
@ -1472,6 +1491,7 @@ class Text(Record, GeometryMixin):
x: X-offset. Default `None` (use modal). x: X-offset. Default `None` (use modal).
y: Y-offset. Default `None` (use modal). y: Y-offset. Default `None` (use modal).
repetition: Repetition. Default `None` (no repetition). repetition: Repetition. Default `None` (no repetition).
properties: List of property records associated with this record.
""" """
self.layer = layer self.layer = layer
self.datatype = datatype self.datatype = datatype
@ -1482,6 +1502,7 @@ class Text(Record, GeometryMixin):
self.string = string self.string = string
else: else:
self.string = AString(string) self.string = AString(string)
self.properties = [] if properties is None else properties
def get_string(self) -> Union[AString, int]: def get_string(self) -> Union[AString, int]:
return verify_modal(self.string) # type: ignore return verify_modal(self.string) # type: ignore
@ -1575,6 +1596,7 @@ class Rectangle(Record, GeometryMixin):
y (Optional[int]): y-offset of the rectangle's lower-left (min-y) point. y (Optional[int]): y-offset of the rectangle's lower-left (min-y) point.
None means reuse modal None means reuse modal
repetition (Optional[repetition_t]): Repetition, if any. repetition (Optional[repetition_t]): Repetition, if any.
properties (List[Property]): List of property records associate with this record.
""" """
layer: Optional[int] = None layer: Optional[int] = None
datatype: Optional[int] = None datatype: Optional[int] = None
@ -1584,6 +1606,7 @@ class Rectangle(Record, GeometryMixin):
y: Optional[int] = None y: Optional[int] = None
repetition: Optional[repetition_t] = None repetition: Optional[repetition_t] = None
is_square: bool = False is_square: bool = False
properties: List['Property']
def __init__(self, def __init__(self,
is_square: bool = False, is_square: bool = False,
@ -1593,7 +1616,8 @@ class Rectangle(Record, GeometryMixin):
height: Optional[int] = None, height: Optional[int] = None,
x: Optional[int] = None, x: Optional[int] = None,
y: Optional[int] = None, y: Optional[int] = None,
repetition: Optional[repetition_t] = None): repetition: Optional[repetition_t] = None,
properties: Optional[List['Property']] = None):
self.is_square = is_square self.is_square = is_square
self.layer = layer self.layer = layer
self.datatype = datatype self.datatype = datatype
@ -1604,6 +1628,7 @@ class Rectangle(Record, GeometryMixin):
self.repetition = repetition self.repetition = repetition
if is_square and self.height is not None: if is_square and self.height is not None:
raise InvalidDataError('Rectangle is square and also has height') raise InvalidDataError('Rectangle is square and also has height')
self.properties = [] if properties is None else properties
def get_width(self) -> int: def get_width(self) -> int:
return verify_modal(self.width) return verify_modal(self.width)
@ -1710,6 +1735,7 @@ class Polygon(Record, GeometryMixin):
None means reuse modal None means reuse modal
repetition (Optional[repetition_t]): Repetition, if any. repetition (Optional[repetition_t]): Repetition, if any.
Default no repetition. Default no repetition.
properties (List[Property]): List of property records associate with this record.
""" """
layer: Optional[int] = None layer: Optional[int] = None
datatype: Optional[int] = None datatype: Optional[int] = None
@ -1717,6 +1743,7 @@ class Polygon(Record, GeometryMixin):
y: Optional[int] = None y: Optional[int] = None
repetition: Optional[repetition_t] = None repetition: Optional[repetition_t] = None
point_list: Optional[point_list_t] = None point_list: Optional[point_list_t] = None
properties: List['Property']
def __init__(self, def __init__(self,
point_list: Optional[point_list_t] = None, point_list: Optional[point_list_t] = None,
@ -1724,13 +1751,15 @@ class Polygon(Record, GeometryMixin):
datatype: Optional[int] = None, datatype: Optional[int] = None,
x: Optional[int] = None, x: Optional[int] = None,
y: Optional[int] = None, y: Optional[int] = None,
repetition: Optional[repetition_t] = None): repetition: Optional[repetition_t] = None,
properties: Optional[List['Property']] = None):
self.layer = layer self.layer = layer
self.datatype = datatype self.datatype = datatype
self.x = x self.x = x
self.y = y self.y = y
self.repetition = repetition self.repetition = repetition
self.point_list = point_list self.point_list = point_list
self.properties = [] if properties is None else properties
if point_list is not None: if point_list is not None:
if len(point_list) < 3: if len(point_list) < 3:
@ -1829,6 +1858,7 @@ class Path(Record, GeometryMixin):
x (Optional[int]): x-offset, None means reuse modal x (Optional[int]): x-offset, None means reuse modal
y (Optional[int]): y-offset, None means reuse modal y (Optional[int]): y-offset, None means reuse modal
repetition (Optional[repetition_t]): Repetition, if any repetition (Optional[repetition_t]): Repetition, if any
properties (List[Property]): List of property records associate with this record.
""" """
layer: Optional[int] = None layer: Optional[int] = None
datatype: Optional[int] = None datatype: Optional[int] = None
@ -1839,6 +1869,7 @@ class Path(Record, GeometryMixin):
half_width: Optional[int] = None half_width: Optional[int] = None
extension_start: Optional[pathextension_t] = None extension_start: Optional[pathextension_t] = None
extension_end: Optional[pathextension_t] = None extension_end: Optional[pathextension_t] = None
properties: List['Property']
def __init__(self, def __init__(self,
point_list: Optional[point_list_t] = None, point_list: Optional[point_list_t] = None,
@ -1849,7 +1880,8 @@ class Path(Record, GeometryMixin):
datatype: Optional[int] = None, datatype: Optional[int] = None,
x: Optional[int] = None, x: Optional[int] = None,
y: Optional[int] = None, y: Optional[int] = None,
repetition: Optional[repetition_t] = None): repetition: Optional[repetition_t] = None,
properties: Optional[List['Property']] = None):
self.layer = layer self.layer = layer
self.datatype = datatype self.datatype = datatype
self.x = x self.x = x
@ -1859,6 +1891,7 @@ class Path(Record, GeometryMixin):
self.half_width = half_width self.half_width = half_width
self.extension_start = extension_start self.extension_start = extension_start
self.extension_end = extension_end self.extension_end = extension_end
self.properties = [] if properties is None else properties
def get_point_list(self) -> point_list_t: def get_point_list(self) -> point_list_t:
return verify_modal(self.point_list) return verify_modal(self.point_list)
@ -2006,6 +2039,7 @@ class Trapezoid(Record, GeometryMixin):
y (Optional[int]): y-offset to lower-left corner of the trapezoid's bounding box. y (Optional[int]): y-offset to lower-left corner of the trapezoid's bounding box.
None means reuse modal None means reuse modal
repetition (Optional[repetition_t]): Repetition, if any repetition (Optional[repetition_t]): Repetition, if any
properties (List[Property]): List of property records associate with this record.
""" """
layer: Optional[int] = None layer: Optional[int] = None
datatype: Optional[int] = None datatype: Optional[int] = None
@ -2017,6 +2051,7 @@ class Trapezoid(Record, GeometryMixin):
delta_a: int = 0 delta_a: int = 0
delta_b: int = 0 delta_b: int = 0
is_vertical: bool is_vertical: bool
properties: List['Property']
def __init__(self, def __init__(self,
is_vertical: bool, is_vertical: bool,
@ -2028,7 +2063,8 @@ class Trapezoid(Record, GeometryMixin):
height: int = None, height: int = None,
x: int = None, x: int = None,
y: int = None, y: int = None,
repetition: repetition_t = None): repetition: repetition_t = None,
properties: Optional[List['Property']] = None):
""" """
Raises: Raises:
InvalidDataError: if dimensions are impossible. InvalidDataError: if dimensions are impossible.
@ -2043,6 +2079,7 @@ class Trapezoid(Record, GeometryMixin):
self.x = x self.x = x
self.y = y self.y = y
self.repetition = repetition self.repetition = repetition
self.properties = [] if properties is None else properties
if self.is_vertical: if self.is_vertical:
if height is not None and delta_b - delta_a > height: if height is not None and delta_b - delta_a > height:
@ -2206,6 +2243,7 @@ class CTrapezoid(Record, GeometryMixin):
y (Optional[int]): y-offset of lower-left (min-y) point of bounding box. y (Optional[int]): y-offset of lower-left (min-y) point of bounding box.
None means reuse modal None means reuse modal
repetition (Optional[repetition_t]): Repetition, if any repetition (Optional[repetition_t]): Repetition, if any
properties (List[Property]): List of property records associate with this record.
""" """
ctrapezoid_type: Optional[int] = None ctrapezoid_type: Optional[int] = None
layer: Optional[int] = None layer: Optional[int] = None
@ -2215,6 +2253,7 @@ class CTrapezoid(Record, GeometryMixin):
x: Optional[int] = None x: Optional[int] = None
y: Optional[int] = None y: Optional[int] = None
repetition: Optional[repetition_t] = None repetition: Optional[repetition_t] = None
properties: List['Property']
def __init__(self, def __init__(self,
ctrapezoid_type: int = None, ctrapezoid_type: int = None,
@ -2224,7 +2263,8 @@ class CTrapezoid(Record, GeometryMixin):
height: int = None, height: int = None,
x: int = None, x: int = None,
y: int = None, y: int = None,
repetition: repetition_t = None): repetition: repetition_t = None,
properties: Optional[List['Property']] = None):
""" """
Raises: Raises:
InvalidDataError: if dimensions are invalid. InvalidDataError: if dimensions are invalid.
@ -2237,6 +2277,7 @@ class CTrapezoid(Record, GeometryMixin):
self.x = x self.x = x
self.y = y self.y = y
self.repetition = repetition self.repetition = repetition
self.properties = [] if properties is None else properties
self.check_valid() self.check_valid()
@ -2405,6 +2446,7 @@ class Circle(Record, GeometryMixin):
x (Optional[int]): x-offset, None means reuse modal x (Optional[int]): x-offset, None means reuse modal
y (Optional[int]): y-offset, None means reuse modal y (Optional[int]): y-offset, None means reuse modal
repetition (Optional[repetition_t]): Repetition, if any repetition (Optional[repetition_t]): Repetition, if any
properties (List[Property]): List of property records associate with this record.
""" """
layer: Optional[int] = None layer: Optional[int] = None
datatype: Optional[int] = None datatype: Optional[int] = None
@ -2412,6 +2454,7 @@ class Circle(Record, GeometryMixin):
y: Optional[int] = None y: Optional[int] = None
repetition: Optional[repetition_t] = None repetition: Optional[repetition_t] = None
radius: Optional[int] = None radius: Optional[int] = None
properties: List['Property']
def __init__(self, def __init__(self,
radius: int = None, radius: int = None,
@ -2419,7 +2462,8 @@ class Circle(Record, GeometryMixin):
datatype: int = None, datatype: int = None,
x: int = None, x: int = None,
y: int = None, y: int = None,
repetition: repetition_t = None): repetition: repetition_t = None,
properties: Optional[List['Property']] = None):
""" """
Args: Args:
radius: Radius. Default `None` (reuse modal). radius: Radius. Default `None` (reuse modal).
@ -2428,6 +2472,7 @@ class Circle(Record, GeometryMixin):
x: X-offset. Default `None` (use modal). x: X-offset. Default `None` (use modal).
y: Y-offset. Default `None` (use modal). y: Y-offset. Default `None` (use modal).
repetition: Repetition. Default `None` (no repetition). repetition: Repetition. Default `None` (no repetition).
properties: List of property records associated with this record.
Raises: Raises:
InvalidDataError: if dimensions are invalid. InvalidDataError: if dimensions are invalid.
@ -2438,6 +2483,7 @@ class Circle(Record, GeometryMixin):
self.x = x self.x = x
self.y = y self.y = y
self.repetition = repetition self.repetition = repetition
self.properties = [] if properties is None else properties
def get_radius(self) -> int: def get_radius(self) -> int:
return verify_modal(self.radius) return verify_modal(self.radius)