22__all__ = [
"AmplifierGeometryComparison",
"ReadoutCornerValNameDict",
"ReadoutCornerNameValDict"]
28from ._cameraGeom
import Amplifier, ReadoutCorner
31ReadoutCornerValNameDict = {
32 ReadoutCorner.LL:
"LL",
33 ReadoutCorner.LR:
"LR",
34 ReadoutCorner.UR:
"UR",
35 ReadoutCorner.UL:
"UL",
37ReadoutCornerNameValDict = {val: key
for key, val
in
38 ReadoutCornerValNameDict.items()}
42 """Flags used to report geometric differences between amplifier"""
45 """All tested properties of the two amplifiers are equal."""
47 SHIFTED_X = enum.auto()
48 """Amplifiers have different X offsets relative to assembled raw."""
50 SHIFTED_Y = enum.auto()
51 """Amplifiers have different Y offsets relative to assembled raw."""
53 SHIFTED = SHIFTED_X | SHIFTED_Y
54 """Amplifiers are different offsets relative to assembled raw."""
56 FLIPPED_X = enum.auto()
57 """Amplifiers differ by (at least) an X-coordinate flip."""
59 FLIPPED_Y = enum.auto()
60 """Amplifiers differ by (at least) a Y-coordinate flip."""
62 FLIPPED = FLIPPED_X | FLIPPED_Y
63 """Amplifiers differ by (at least) a coordinate flip."""
65 ASSEMBLY_DIFFERS = SHIFTED | FLIPPED
66 """Amplifiers differ in offsets relative to raw, indicating at least a
67 difference in assembly state.
70 REGIONS_DIFFER = enum.auto()
71 """Amplifiers have different full/data/overscan/prescan regions.
73 If ``assembly=True`` was passed to `Amplifier.compare`, this will only be
74 set
if regions differ even after applying flips
and offsets to make the
75 assembly states the same. If ``assembly=
False`` was passed to
76 `Amplifier.compare`, regions will be compared
while assuming that assembly
85 """Compare the geometry of this amplifier with another.
89 assembly : `bool`, optional
90 If `True` (default) test whether flips
and offsets relative to
91 assembled raw are the same,
and account
for those when testing
92 whether regions are the same.
93 regions : `bool`, optional
94 If `
True` (default) test whether full/data/overscan/prescan regions
99 comparison : `AmplifierGeometryComparison`
100 Flags representing the result of the comparison.
102 result = AmplifierGeometryComparison.EQUAL
104 if self.getRawXYOffset().getX() != other.getRawXYOffset().getX():
105 result |= AmplifierGeometryComparison.SHIFTED_X
106 if self.getRawXYOffset().getY() != other.getRawXYOffset().getY():
107 result |= AmplifierGeometryComparison.SHIFTED_Y
108 if self.getRawFlipX() != other.getRawFlipX():
109 result |= AmplifierGeometryComparison.FLIPPED_X
110 if self.getRawFlipY() != other.getRawFlipY():
111 result |= AmplifierGeometryComparison.FLIPPED_Y
113 if result & AmplifierGeometryComparison.ASSEMBLY_DIFFERS:
117 outOffset=self.getRawXYOffset(),
118 outFlipX=self.getRawFlipX(),
119 outFlipY=self.getRawFlipY(),
122 "HorizontalOverscan",
126 if getattr(self, f
"getRaw{bboxName}BBox")() != getattr(other, f
"getRaw{bboxName}BBox")():
127 result |= AmplifierGeometryComparison.REGIONS_DIFFER
131@inClass(Amplifier.Builder)
132def transform(self, *, outOffset=None, outFlipX=False, outFlipY=False):
133 """Transform an amplifier builder (in-place) by applying shifts and
139 Post-transformation return value
for ``self.getRawXYOffset()``.
140 The default
is ``(0, 0)``, which shifts the amplifier to its
141 position
in the assembled (but still untrimmed) raw image.
142 outFlipX : `bool`, optional
143 Post-transformation
return value
for ``self.getRawFlipX()``. The
144 default
is `
False`, which flips the amplifier to its correct
145 X orientation
in the assembled raw image.
146 outFlipX : `bool`, optional
147 Post-transformation
return value
for ``self.getRawFlipY()``. The
148 default
is `
False`, which flips the amplifier to its correct
149 Y orientation
in the assembled raw image.
153 self : `AmplifierBuilder`
154 Returned to enable method chaining, e.g.
155 ``amplifier.rebuild().
transform().finish()``.
157 if outOffset
is None:
158 outOffset = Extent2I(0, 0)
159 bbox = self.getRawBBox()
160 awidth, aheight = bbox.getDimensions()
164 boxMin0 = bbox.getMin()
165 if self.getRawFlipX() != outFlipX:
167 if self.getRawFlipY() != outFlipY:
169 shift = boxMin0 - bbox.getMin()
172 "HorizontalOverscan",
176 bbox = getattr(self, f
"getRaw{bboxName}BBox")()
177 if self.getRawFlipX() != outFlipX:
179 if self.getRawFlipY() != outFlipY:
181 bbox.shift(self.getRawXYOffset() + shift - outOffset)
183 getattr(self, f
"setRaw{bboxName}BBox")(bbox)
186 outReadoutCorner = self.getReadoutCorner()
187 if self.getRawFlipX() != outFlipX:
188 xFlipMapping = {ReadoutCorner.LL: ReadoutCorner.LR, ReadoutCorner.LR: ReadoutCorner.LL,
189 ReadoutCorner.UR: ReadoutCorner.UL, ReadoutCorner.UL: ReadoutCorner.UR}
190 outReadoutCorner = xFlipMapping[outReadoutCorner]
191 if self.getRawFlipY() != outFlipY:
192 yFlipMapping = {ReadoutCorner.LL: ReadoutCorner.UL, ReadoutCorner.LR: ReadoutCorner.UR,
193 ReadoutCorner.UR: ReadoutCorner.LR, ReadoutCorner.UL: ReadoutCorner.LL}
194 outReadoutCorner = yFlipMapping[outReadoutCorner]
195 if outReadoutCorner != self.getReadoutCorner():
196 self.setReadoutCorner(outReadoutCorner)
201 self.setRawXYOffset(outOffset)
202 self.setRawFlipX(outFlipX)
203 self.setRawFlipY(outFlipY)
compareGeometry(self, other, *assembly=True, regions=True)