Table of Contents
Name | Direction | Type | Default | Description |
---|---|---|---|---|
Workspace | InOut | Workspace | Mandatory | The name of the workspace for which the new instrument configuration will have an effect. Any other workspaces stored in the analysis data service will be unaffected. |
ComponentName | Input | string | The name of the component to rotate. Component names are defined in the instrument definition files. | |
DetectorID | Input | number | -1 | The ID of the detector to rotate. If both the component name and the detector ID are set the latter will be used. |
X | Input | number | 0 | The x-part of the rotation axis. |
Y | Input | number | 0 | The y-part of the rotation axis. |
Z | Input | number | 0 | The z-part of the rotation axis. |
Angle | Input | number | 0 | The angle of rotation in degrees. |
RelativeRotation | Input | boolean | True | The property defining how the rotation should be interpreted. If true it is a relative rotation. Otherwise it is an absolute rotation. |
RotateInstrumentComponent rotates a component around an axis of rotation by an angle given in degrees. Rotation by 0 degrees does not change the component’s orientation. The rotation axis (X,Y,Z) must be given in the co-ordinate system attached to the component and rotates with it. The centre of rotation is the centre of the component to be rotated, and any children of that component (such as detectors in a bank, or pixels in a tube) are moved and rotated along with the component.
# Load a MUSR file
musr = Load('MUSR00015189')
# and use the first workspace in the workspace group
ws = mtd['musr_1']
def pos3D_as_str(pos, digits=6):
""" Produces a string with a V3D position (x, y, z) from a V3D object,
using a fixed limited number of digits (for robust string comparisons).
"""
def nz(value):
""" Handles potential issues with +-0 (for 6 digits text output) """
return 0.0 if abs(value) < 1e-7 else value
precision = str(digits)
format_str = '[{0:.'+precision+'f}, {1:.'+precision+'f}, {2:.'+precision+'f}]'
result = format_str.format(nz(pos.getX()), nz(pos.getY()), nz(pos.getZ()))
return result
print('Original positions of detectors 1 and 2')
opos1 = ws.getInstrument().getDetector(1).getPos()
opos2 = ws.getInstrument().getDetector(2).getPos()
print('Det 1: {0}'.format(pos3D_as_str(opos1)))
print('Det 2: {0}'.format(pos3D_as_str(opos2)))
# Rotate bank 'back' around the Z axis by 90
RotateInstrumentComponent( ws, ComponentName='back', X=0,Y=1,Z=0, Angle=90.0 )
print('Positions of detectors 1 and 2 after rotation')
pos1 = ws.getInstrument().getDetector(1).getPos()
pos2 = ws.getInstrument().getDetector(2).getPos()
print('Det 1: {0}'.format(pos3D_as_str(pos1)))
print('Det 2: {0}'.format(pos3D_as_str(pos2)))
# Load a MUSR file
musr = Load('MUSR00015189')
# and use the first workspace in the workspace group
ws = mtd['musr_1']
def pos3D_as_str(pos, digits=6):
""" Produces a string with a V3D position (x, y, z) from a V3D object,
using a fixed limited number of digits (for robust string comparisons).
"""
def nz(value):
""" Handles potential issues with +-0 (for 6 digits text output) """
return 0.0 if abs(value) < 1e-7 else value
precision = str(digits)
format_str = '[{0:.'+precision+'f}, {1:.'+precision+'f}, {2:.'+precision+'f}]'
result = format_str.format(nz(pos.getX()), nz(pos.getY()), nz(pos.getZ()))
return result
print('Original positions of detectors 1 and 4')
opos1 = ws.getInstrument().getDetector(1).getPos()
opos4 = ws.getInstrument().getDetector(4).getPos()
print('Det 1: {0}'.format(pos3D_as_str(opos1)))
print('Det 4: {0}'.format(pos3D_as_str(opos4)))
# Rotate bank 'back' around the Z axis by 3 detectors.
RotateInstrumentComponent( ws, ComponentName='back', X=0,Y=0,Z=1, Angle=3*360.0 / 32 )
print('Positions of detector 1 after rotation')
pos1 = ws.getInstrument().getDetector(1).getPos()
print('Det 1: {0}'.format(pos3D_as_str(pos1)))
print('Detector 1 took place of detector 4')
import numpy as np
# Load a MUSR file
musr = Load('MUSR00015189')
# and use the first workspace in the workspace group
ws = mtd['musr_1']
# Rotating a detector doesn't change its position, just its orientation
# Original position of detector 33
print(ws.getInstrument().getDetector(33).getPos())
# Caclulate the solid angles for all detectors in the instrument
# The result is a single-bin workspace with solid angles for all spectra in ws
saws = SolidAngle( ws )
# Collect the solid angles from the first bin in saws and save them in numpy array.
# Numpy module makes it easy to manipulate arrays
sa1 = np.array( [ saws.readY(i)[0] for i in range(saws.getNumberHistograms()) ] )
# Rotate detector 33 around the Z axis by 90 degrees.
RotateInstrumentComponent( ws, DetectorID=33, X=0,Y=0,Z=1, Angle=90 )
# Check the position of detector 33 stays unchanged
print(ws.getInstrument().getDetector(33).getPos())
# Calculate the solid angles after rotation
saws = SolidAngle( ws )
sa2 = np.array( [ saws.readY(i)[0] for i in range(saws.getNumberHistograms()) ] )
# Take element by element difference of the solid angles
diff = sa2 - sa1
print(diff)
print('The non-zero difference {:.13f} is due to detector {}'.format(diff[32], ws.getDetector(32).getID()))
[0.0888151,-0.108221,-0.145]
[0.0888151,-0.108221,-0.145]
[ 0. 0. 0. 0. 0. 0. 0.
0. 0. 0. 0. 0. 0. 0.
0. 0. 0. 0. 0. 0. 0.
0. 0. 0. 0. 0. 0. 0.
0. 0. 0. 0. -0.04645313 0. 0.
0. 0. 0. 0. 0. 0. 0.
0. 0. 0. 0. 0. 0. 0.
0. 0. 0. 0. 0. 0. 0.
0. 0. 0. 0. 0. 0. 0.
0. ]
The non-zero difference -0.0464531276188 is due to detector 33
Categories: Algorithms | DataHandling\Instrument
C++ source: RotateInstrumentComponent.cpp (last modified: 2018-03-07)
C++ header: RotateInstrumentComponent.h (last modified: 2018-03-07)