-
Notifications
You must be signed in to change notification settings - Fork 10
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
adding basic functionality for KDC101 KCube. #162
Open
edyoshikun
wants to merge
23
commits into
main
Choose a base branch
from
thorlabs_stage
base: main
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
Open
Changes from all commits
Commits
Show all changes
23 commits
Select commit
Hold shift + click to select a range
809a891
adding basic functionality for KDC101 KCube.
edyoshikun 9e81839
adding the KIM001 piezo inertia stage
edyoshikun 22b6838
updating abstract stage class Implement the abstract class for stage…
edyoshikun 9f66f3d
adding two more options for abtrasct class
edyoshikun a2d6a08
updating simulated KIM001 intertial motor
edyoshikun c667680
changing the dc motor to add abstract functions
edyoshikun 5a2c5cf
no need to zero after disconnecting - Ed's commit from mantis
ieivanov 3ca0707
adding context manager and changing the default acceleration values a…
ieivanov 1629cf2
context manager for the KDC101
ieivanov ed6d794
context manager for the KDC101 - ED Hirata
ieivanov b1f42d8
Merge branch 'thorlabs_stage' of https://github.com/czbiohub/coPylot …
ieivanov 57c5ac3
adding pythonnet required to read the dlls from thorlabs
edyoshikun dc5516d
add stage `move_absolute` method
ieivanov 016e097
update KIM101 url
ieivanov cc7bf66
define move_absolute in child classes
ieivanov 65f160a
remove delay from KIM001 position setter
ieivanov a61a5ce
update list_available_stages
ieivanov 3d959a0
raise error on position outside of travel range
ieivanov 195ef17
enforce that position is int32
ieivanov 16dadf1
set device name and fix int32 bug
ieivanov 7f4be85
enforce more int casting
ieivanov 872ae42
fix bug with relative move
ieivanov 407f7a1
move check for 0 moves to position setter
ieivanov File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,59 @@ | ||
from abc import ABCMeta, abstractmethod | ||
|
||
|
||
class AbstractStage(metaclass=ABCMeta): | ||
|
||
@staticmethod | ||
@abstractmethod | ||
def list_available_stages(): | ||
"""List all stages that driver discovers.""" | ||
raise NotImplementedError() | ||
|
||
@property | ||
@abstractmethod | ||
def position(self): | ||
"Method to get/set the position in um" | ||
raise NotImplementedError() | ||
|
||
@position.setter | ||
@abstractmethod | ||
def position(self, value): | ||
raise NotImplementedError() | ||
|
||
@property | ||
@abstractmethod | ||
def travel_range(self): | ||
""" | ||
Valid minimum and maximum travel range values. | ||
Returns | ||
------- | ||
Tuple | ||
(min_valid_position, max_valid_position) | ||
""" | ||
raise NotImplementedError() | ||
|
||
@travel_range.setter | ||
@abstractmethod | ||
def travel_range(self, value): | ||
""" | ||
Set the travel range of the stage | ||
---- | ||
Tuple | ||
(min_valid_position, max_valid_position) | ||
""" | ||
raise NotImplementedError() | ||
|
||
@abstractmethod | ||
def move_relative(self, value): | ||
"Move a relative distance from current position" | ||
raise NotImplementedError() | ||
|
||
@abstractmethod | ||
def move_absolute(self, value): | ||
"Move to an absolute position" | ||
raise NotImplementedError() | ||
|
||
@abstractmethod | ||
def zero_position(self): | ||
"Move the stage to 0 position" | ||
raise NotImplementedError() |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,189 @@ | ||
""" | ||
Thorlabs KDC101 K-Cube for DC Motors wrapper | ||
|
||
The user must install Thorlabs Kinesis and reference the appropriate paths below | ||
|
||
For more details: | ||
https://github.com/Thorlabs/Motion_Control_Examples/blob/main/Python/KCube/KDC101/kdc101_pythonnet.py | ||
|
||
""" | ||
import os | ||
import time | ||
import sys | ||
import clr | ||
import os | ||
from copylot import logger | ||
from copylot.hardware.stages.abstract_stage import AbstractStage | ||
import time | ||
|
||
clr.AddReference( | ||
"C:\\Program Files\\Thorlabs\\Kinesis\\Thorlabs.MotionControl.DeviceManagerCLI.dll" | ||
) | ||
clr.AddReference( | ||
"C:\\Program Files\\Thorlabs\\Kinesis\\Thorlabs.MotionControl.GenericMotorCLI.dll" | ||
) | ||
clr.AddReference( | ||
"C:\\Program Files\\Thorlabs\\Kinesis\\ThorLabs.MotionControl.KCube.DCServoCLI.dll" | ||
) | ||
|
||
from Thorlabs.MotionControl.DeviceManagerCLI import * | ||
from Thorlabs.MotionControl.GenericMotorCLI import * | ||
from Thorlabs.MotionControl.KCube.DCServoCLI import * | ||
from System import Decimal | ||
|
||
|
||
class KCube_DCServo(AbstractStage): | ||
def __init__( | ||
self, | ||
device_name, | ||
serial_number='27000001', | ||
stage_positive='forward', | ||
polling=False, | ||
simulator=False, | ||
): | ||
self.serial_number = serial_number | ||
self.device = None | ||
self.device_name = device_name | ||
self.simulator = simulator | ||
self.timeout = 20000 # ms | ||
self.device_config = None | ||
self.stage_direction = stage_positive | ||
self.polling = polling | ||
self.max_travel_range = None | ||
self.min_travel_range = None | ||
|
||
if self.simulator: | ||
SimulationManager.Instance.InitializeSimulations() | ||
|
||
self.device_list() | ||
self.connect() | ||
|
||
def __enter__(self): | ||
return self | ||
|
||
def __exit__(self, exc_type, exc_val, exc_tb): | ||
self.close() | ||
logger.info("thorlabs stage disconnected") | ||
|
||
def list_available_stages(self): | ||
DeviceManagerCLI.BuildDeviceList() | ||
dev_list = DeviceManagerCLI.GetDeviceList() | ||
logger.info(f"Device List {dev_list}") | ||
return DeviceManagerCLI.GetDeviceList() | ||
|
||
def load_configuration(self): | ||
self.device_config = self.device.LoadMotorConfiguration( | ||
self.serial_number, | ||
DeviceConfiguration.DeviceSettingsUseOptionType.UseFileSettings, | ||
) | ||
self.device_config.DeviceSettingsName = self.device_name | ||
self.device_config.UpdateCurrentConfiguration() | ||
self.device.SetSettings(self.device.MotorDeviceSettings, True, False) | ||
|
||
def connect(self): | ||
if self._is_initialized(): | ||
logger.info("Device is already initialized") | ||
else: | ||
logger.info(f"Initializing device with serial number {self.serial_number}") | ||
|
||
if self.serial_number is not None: | ||
self.device = KCubeDCServo.CreateKCubeDCServo(self.serial_number) | ||
self.device.Connect(self.serial_number) | ||
time.sleep(0.25) | ||
if self.polling: | ||
# Start polling and enable channel | ||
# Polling blocks unless spawn the device in a separate thread | ||
self.device.StartPolling(250) # 250ms polling rate | ||
time.sleep(25) | ||
self.device.EnableDevice() | ||
time.sleep(0.25) # Wait for device to enable | ||
|
||
self.device_info() | ||
self._is_initialized() | ||
self.load_configuration() | ||
|
||
def _is_initialized(self): | ||
# Ensure that the device settings have been initialized | ||
if self.device is not None: | ||
if not self.device.IsSettingsInitialized(): | ||
self.device.WaitForSettingsInitialized(10000) # 10 second timeout | ||
assert self.device.IsSettingsInitialized() is True | ||
else: | ||
return False | ||
|
||
def device_info(self): | ||
# Get Device Information and display description | ||
device_info = self.device.GetDeviceInfo() | ||
logger.info(device_info.Description) | ||
return device_info.Description | ||
|
||
def zero_position(self): | ||
logger.info('Zeroing device') | ||
self.device.SetPositionAs(self.channel, 0) | ||
|
||
def close(self): | ||
# TODO: does it need to move back to a position? | ||
# self.position = 0.0 | ||
if self.polling: | ||
self.device.StopPolling() | ||
self.device.Disconnect() | ||
|
||
if self.simulator: | ||
SimulationManager.Instance.UninitializeSimulations() | ||
|
||
def is_initialized(self): | ||
if not self.device.IsSettingsInitialized(): | ||
self.device.WaitForSettingsInitialized(10000) # 10 second timeout | ||
assert self.device.IsSettingsInitialized() is True | ||
|
||
def device_info(self): | ||
self.device_info = self.device.GetDeviceInfo() | ||
logger.info(self.device_info.Description) | ||
|
||
@property | ||
def position(self): | ||
return float(str(self.device.Position)) | ||
|
||
@position.setter | ||
def position(self, value): | ||
if self.min_travel_range and self.max_travel_range is not None: | ||
if value > self.max_travel_range: | ||
value = self.max_travel_range | ||
if value < self.min_travel_range: | ||
value = self.min_travel_range | ||
value = Decimal(value) | ||
self.device.MoveTo(value, self.timeout) | ||
time.sleep(1) | ||
logger.info(f'Stage< {self.device_name} > reached position: {value}') | ||
|
||
def move_relative(self, value): | ||
abs_value = abs(value) | ||
if self.stage_direction == 'forward': | ||
if value > 0: | ||
(MotorDirection.Forward, Decimal(abs_value), self.timeout) | ||
else: | ||
(MotorDirection.Backward, Decimal(abs_value), self.timeout) | ||
else: | ||
if value > 0: | ||
(MotorDirection.Backward, Decimal(abs_value), self.timeout) | ||
else: | ||
(MotorDirection.Forward, Decimal(abs_value), self.timeout) | ||
|
||
def move_relative_2(self, value): | ||
target_position = self.position + value | ||
self.position = target_position | ||
|
||
def move_absolute(self, value): | ||
self.position = value | ||
|
||
def home_device(self): | ||
return self.device.Home(60000) | ||
|
||
@property | ||
def travel_range(self): | ||
return (self.min_travel_range, self.max_travel_range) | ||
|
||
@travel_range.setter | ||
def travel_range(self, value): | ||
self.max_travel_range = value[0] | ||
self.min_travel_range = value[1] |
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
RENAME TO LIMITS