diff --git a/.eslintrc.js b/.eslintrc.js index 3d54b0c8fe9..4528c2fed2d 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -13,6 +13,7 @@ module.exports = { 'prettier', 'plugin:json/recommended', 'plugin:storybook/recommended', + 'plugin:react/jsx-runtime', ], plugins: ['react', 'react-hooks', 'json', 'testing-library'], diff --git a/.github/workflows/opentrons-ai-server-lint-test.yaml b/.github/workflows/opentrons-ai-server-lint-test.yaml index d1a169b024d..8fe9bcb19ea 100644 --- a/.github/workflows/opentrons-ai-server-lint-test.yaml +++ b/.github/workflows/opentrons-ai-server-lint-test.yaml @@ -27,7 +27,7 @@ jobs: - name: Setup Python uses: 'actions/setup-python@v5' with: - python-version: '3.12.4' + python-version: '3.12' cache: 'pipenv' cache-dependency-path: opentrons-ai-server/Pipfile.lock - name: Setup diff --git a/.storybook/preview.jsx b/.storybook/preview.jsx index 4d97efe9c4d..bdae9a4f833 100644 --- a/.storybook/preview.jsx +++ b/.storybook/preview.jsx @@ -1,4 +1,3 @@ -import React from 'react' import { I18nextProvider } from 'react-i18next' import { GlobalStyle } from '../app/src/atoms/GlobalStyle' import { i18n } from '../app/src/i18n' diff --git a/api/docs/img/partial-pickup-deck-extents.png b/api/docs/img/partial-pickup-deck-extents.png new file mode 100644 index 00000000000..9977357139a Binary files /dev/null and b/api/docs/img/partial-pickup-deck-extents.png differ diff --git a/api/docs/v2/pipettes/characteristics.rst b/api/docs/v2/pipettes/characteristics.rst index 8aaf8751dd0..9203dd81816 100644 --- a/api/docs/v2/pipettes/characteristics.rst +++ b/api/docs/v2/pipettes/characteristics.rst @@ -158,18 +158,18 @@ These flow rate properties operate independently. This means you can specify dif Let's tell the robot to aspirate, dispense, and blow out the liquid using default flow rates. Notice how you don't need to specify a ``flow_rate`` attribute to use the defaults:: - pipette.aspirate(200, plate["A1"]) # 160 µL/s - pipette.dispense(200, plate["A2"]) # 160 µL/s - pipette.blow_out() # 80 µL/s + pipette.aspirate(200, plate["A1"]) # 716 µL/s + pipette.dispense(200, plate["A2"]) # 716 µL/s + pipette.blow_out() # 716 µL/s Now let's change the flow rates for each action:: pipette.flow_rate.aspirate = 50 pipette.flow_rate.dispense = 100 - pipette.flow_rate.blow_out = 75 + pipette.flow_rate.blow_out = 300 pipette.aspirate(200, plate["A1"]) # 50 µL/s pipette.dispense(200, plate["A2"]) # 100 µL/s - pipette.blow_out() # 75 µL/s + pipette.blow_out() # 300 µL/s These flow rates will remain in effect until you change the ``flow_rate`` attribute again *or* call ``configure_for_volume()``. Calling ``configure_for_volume()`` always resets all pipette flow rates to the defaults for the mode that it sets. @@ -184,7 +184,7 @@ These flow rates will remain in effect until you change the ``flow_rate`` attrib Flex Pipette Flow Rates ----------------------- -The default flow rates for Flex pipettes depend on the maximum volume of the pipette and the capacity of the currently attached tip. For each pipette–tip configuration, the default flow rate is the same for aspirate, dispense, and blowout actions. +Flex pipette flow rates depend on pipette volume and tip capacity. Each pipette–tip combination has a default flow rate for aspirating, dispensing, and blowing out liquid. When using a 50 µL pipette, you should only use 50 µL tips. .. list-table:: :header-rows: 1 @@ -193,7 +193,7 @@ The default flow rates for Flex pipettes depend on the maximum volume of the pip - Tip Capacity (µL) - Flow Rate (µL/s) * - 50 µL (1- and 8-channel) - - All capacities + - 50 - 57 * - 1000 µL (1-, 8-, and 96-channel) - 50 diff --git a/api/docs/v2/pipettes/partial_tip_pickup.rst b/api/docs/v2/pipettes/partial_tip_pickup.rst index e799d25554f..9890cc7d345 100644 --- a/api/docs/v2/pipettes/partial_tip_pickup.rst +++ b/api/docs/v2/pipettes/partial_tip_pickup.rst @@ -183,10 +183,10 @@ The ``start`` parameter sets the first and only nozzle used in the configuration - | Back to front, left to right | (A1 through H1, A2 through H2, …) -Since they follow the same pickup order as a single-channel pipette, Opentrons recommends using the following configurations: +.. warning:: + In certain conditions, tips in adjacent columns may cling to empty nozzles during single-tip pickup. You can avoid this by overriding automatic tip tracking to pick up tips row by row, rather than column by column. The code sample below demonstrates how to pick up tips this way. -- For 8-channel pipettes, ``start="H1"``. -- For 96-channel pipettes, ``start="H12"``. + However, as with all partial tip layouts, be careful that you don't place the pipette in a position where it overlaps more tips than intended. Here is the start of a protocol that imports the ``SINGLE`` and ``ALL`` layout constants, loads an 8-channel pipette, and sets it to pick up a single tip. @@ -210,21 +210,22 @@ Here is the start of a protocol that imports the ``SINGLE`` and ``ALL`` layout c ) pipette.configure_nozzle_layout( style=SINGLE, - start="H12", - tip_racks=[partial_rack] + start="H1" ) .. versionadded:: 2.20 -Since this configuration uses ``start="H12"``, it will pick up tips in the usual order:: +To pick up tips row by row, first construct a list of all wells in the tip rack ordered from A1, A2 … H11, H12. One way to do this is to use :py:func:`sum` to flatten the list of lists returned by :py:meth:`.Labware.rows`:: - pipette.pick_up_tip() # picks up A1 from tip rack - pipette.drop_tip() - pipette.pick_up_tip() # picks up B1 from tip rack + tips_by_row = sum(partial_rack.rows(), []) -.. note:: +Then ``pop`` items from the front of the list (index 0) and pass them as the ``location`` of :py:meth:`.pick_up_tip`:: - You can pick up tips row by row, rather than column by column, by specifying a location for :py:meth:`.pick_up_tip` each time you use it in ``SINGLE`` configuration. However, as with all partial tip layouts, be careful that you don't place the pipette in a position where it overlaps more tips than intended. + # pick up A1 from tip rack + pipette.pick_up_tip(location=tips_by_row.pop(0)) + pipette.drop_tip() + # pick up A2 from tip rack + pipette.pick_up_tip(location=tips_by_row.pop(0)) Partial Column Layout @@ -232,7 +233,7 @@ Partial Column Layout Partial column pickup is available on 8-channel pipettes only. Partial columns contain 2 to 7 consecutive tips in a single column. The pipette always picks up partial columns with its frontmost nozzles (``start="H1"``). -To specify the number of tips to pick up, add the ``end`` parameter when calling :py:meth:`.configure_nozzle_layout`. Use the chart below to determine the end row (G through B) for your desired number of tips. The end column should be the same as your start column (1 or 12). +To specify the number of tips to pick up, add the ``end`` parameter when calling :py:meth:`.configure_nozzle_layout`. Use the chart below to determine the ending nozzle (G1 through B1) for your desired number of tips. .. list-table:: :stub-columns: 1 @@ -244,16 +245,21 @@ To specify the number of tips to pick up, add the ``end`` parameter when calling - 5 - 6 - 7 - * - ``end`` row - - G - - F - - E - - D - - C - - B + * - ``end`` nozzle + - G1 + - F1 + - E1 + - D1 + - C1 + - B1 When picking up 3, 5, 6, or 7 tips, extra tips will be left at the front of each column. You can use these tips with a different nozzle configuration, or you can manually re-rack them at the end of your protocol for future use. +.. warning:: + In certain conditions, tips in adjacent columns may cling to empty nozzles during partial-column pickup. You can avoid this by overriding automatic tip tracking to pick up tips row by row, rather than column by column. The code sample below demonstrates how to pick up tips this way. + + However, as with all partial tip layouts, be careful that you don't place the pipette in a position where it overlaps more tips than intended. + Here is the start of a protocol that imports the ``PARTIAL_COLUMN`` and ``ALL`` layout constants, loads an 8-channel pipette, and sets it to pick up four tips: .. code-block:: python @@ -274,21 +280,24 @@ Here is the start of a protocol that imports the ``PARTIAL_COLUMN`` and ``ALL`` pipette.configure_nozzle_layout( style=PARTIAL_COLUMN, start="H1", - end="E1", - tip_racks=[partial_rack] + end="E1" ) .. versionadded:: 2.20 -This configuration will pick up tips from the back half of column 1, then the front half of column 1, then the back half of column 2, and so on:: +When pipetting in partial column configuration, remember that *the frontmost channel of the pipette is its primary channel*. To pick up tips across the back half of the rack, then across the front half of the rack, construct a list of that includes all and only the wells in row D and row H:: - pipette.pick_up_tip() # picks up A1-D1 from tip rack - pipette.drop_tip() - pipette.pick_up_tip() # picks up E1-H1 from tip rack + tips_by_row = partial_rack.rows_by_name()["D"] + partial_rack.rows_by_name()["H"] + +Then ``pop`` items from the front of the list (index 0) and pass them as the ``location`` of :py:meth:`.pick_up_tip`:: + + # pick up A1-D1 from tip rack + pipette.pick_up_tip(location=tips_by_row.pop(0)) pipette.drop_tip() - pipette.pick_up_tip() # picks up A2-D2 from tip rack + # pick up A2-D2 from tip rack + pipette.pick_up_tip(location=tips_by_row.pop(0)) -When handling liquids in partial column configuration, remember that *the frontmost channel of the pipette is its primary channel*. For example, to use the same configuration as above to transfer liquid from wells A1–D1 to wells A2–D2 on a plate, you must use the wells in row D as the source and destination targets:: +To use the same configuration as above to transfer liquid from wells A1–D1 to wells A2–D2 on a plate, you must use the wells in row D as the source and destination targets:: # pipette in 4-nozzle partial column layout pipette.transfer( @@ -361,14 +370,58 @@ This keeps tip tracking consistent across each type of pickup. And it reduces th Tip Pickup and Conflicts ======================== -During partial tip pickup, pipettes move into spaces above adjacent slots. To avoid crashes, the API prevents you from performing partial tip pickup when there is tall labware in these spaces. The current nozzle layout determines which labware can safely occupy adjacent slots. +During partial tip pickup, the pipette moves into spaces above adjacent slots. To avoid crashes, the API prevents you from performing partial tip pickup in locations where the pipette could collide with the outer edges of the robot or labware in the working area. The current nozzle layout, pickup or pipetting location, and adjacent labware determine whether a particular pipetting action is safe to perform. -The API will raise errors for potential labware crashes when using a partial nozzle configuration. Nevertheless, it's a good idea to do the following when working with partial tip pickup: +The API will raise errors for potential crashes when using a partial nozzle configuration. Nevertheless, it's a good idea to do the following when working with partial tip pickup: - Plan your deck layout carefully. Make a diagram and visualize everywhere the pipette will travel. - - Simulate your protocol and compare the run preview to your expectations of where the pipette will travel. + - Simulate your protocol and compare the output to your expectations of where the pipette will travel. - Perform a dry run with only tip racks on the deck. Have the Emergency Stop Pendant handy in case you see an impending crash. +Deck Extents +------------ + +When using partial nozzle configurations around the back, right, and front edges of the deck, there are limitations on how far the pipette can move beyond the outer edge of the deck slot. The API will raise an error if you try to pipette beyond these outer `extents` of the working area. + +.. tip:: + There are no extents-related limitations on slots B1, B2, C1, and C2. When performing partial pickup and pipetting in these slots, you only have to consider :ref:`possible labware conflicts `. + +One way to think of deck extents is in terms of where you can pick up tips or pipette to a 96-well plate loaded in a given slot. These limitations only apply when using a layout that places the pipette further towards the windows of the robot than an ``ALL`` layout would. For example, using a ``ROW`` layout with the frontmost nozzles of the 96-channel pipette, it will never move farther forward than the H row of a labware in slots D1–D3. But using a ``ROW`` layout with the backmost nozzles would bring it farther forward — it could collide with the front window, except that the API prevents it. + +The following table summarizes the limitations in place along each side of the deck. + +.. list-table:: + :header-rows: 1 + + * - Deck slots + - Nozzle configuration + - Inaccessible wells + * - A1–D1 (left edge) + - Rightmost column + - None (all wells accessible) + * - A1–D3 (back edge) + - Frontmost row + - Rows A–G + * - A3–D3 (right edge) + - Leftmost column + - Columns 11–12 + * - D1–D3 (front edge) + - Backmost row + - Rows F–H + +To visualize these limitations, the below deck map shades all wells that have a single limitation in light blue, and all wells that have two limitations in dark blue. + +.. image:: ../../img/partial-pickup-deck-extents.png + +Multiple limitations occur when you use a ``SINGLE`` configuration that uses the innermost corner nozzle, with respect to the pipette's position on the deck. For example, using nozzle A1 on the 96-channel pipette has multiple limitations in slot D3. + +Additionally, column A of plates loaded on a Thermocycler Module is inaccessible by the rightmost nozzles of the 96-channel pipette. Although the API treats such plates as being in slot A1, the physical location of a plate on the Thermocycler is slightly further left than a plate loaded directly on the slot. + +.. _partial-labware-conflicts: + +Arranging Labware +----------------- + For column pickup, Opentrons recommends using the nozzles in column 12 of the pipette:: pipette.configure_nozzle_layout( diff --git a/api/release-notes.md b/api/release-notes.md index 8664eac3f8d..aad0cadea7a 100644 --- a/api/release-notes.md +++ b/api/release-notes.md @@ -24,6 +24,12 @@ Welcome to the v8.0.0 release of the Opentrons robot software! - Provides more partial tip pickup configurations. All multi-channel pipettes now support single and partial column pickup, and the Flex 96-channel pipette now supports row pickup. - Improves homing behavior when a Flex protocol completes or is canceled with liquid-filled tips attached to the pipette. +### Known Issues + +- During single-tip or partial-column pickup with a multi-channel pipette, tips in adjacent columns may cling to empty nozzles. Pick up tips row by row, rather than column by column, to avoid this. +- Protocol analysis and `opentrons_simulate` do not raise an error when a protocol tries to detect liquid with a pipette nozzle configuration that doesn't contain a pressure sensor (single-tip pickup with A12 or H1). Avoid using the A12 and H1 nozzles for single-tip pickup if you need to detect liquid presence within wells. +- `opentrons_simulate` describes motion to wells only with respect to the primary channel, regardless of the current pipette nozzle configuration. + --- ## Opentrons Robot Software Changes in 7.5.0 diff --git a/api/src/opentrons/drivers/absorbance_reader/abstract.py b/api/src/opentrons/drivers/absorbance_reader/abstract.py index 6dc403fdff9..600497cbd6d 100644 --- a/api/src/opentrons/drivers/absorbance_reader/abstract.py +++ b/api/src/opentrons/drivers/absorbance_reader/abstract.py @@ -1,6 +1,7 @@ from abc import ABC, abstractmethod -from typing import Dict, List, Tuple +from typing import Dict, List, Optional, Tuple from opentrons.drivers.types import ( + ABSMeasurementMode, AbsorbanceReaderLidStatus, AbsorbanceReaderDeviceState, AbsorbanceReaderPlatePresence, @@ -32,11 +33,18 @@ async def get_available_wavelengths(self) -> List[int]: ... @abstractmethod - async def get_single_measurement(self, wavelength: int) -> List[float]: + async def initialize_measurement( + self, + wavelengths: List[int], + mode: ABSMeasurementMode = ABSMeasurementMode.SINGLE, + reference_wavelength: Optional[int] = None, + ) -> None: + """Initialize measurement for the device in single or multi mode for the given wavelengths""" ... @abstractmethod - async def initialize_measurement(self, wavelength: int) -> None: + async def get_measurement(self) -> List[List[float]]: + """Gets one or more measurements based on the current configuration.""" ... @abstractmethod diff --git a/api/src/opentrons/drivers/absorbance_reader/async_byonoy.py b/api/src/opentrons/drivers/absorbance_reader/async_byonoy.py index 6f3ab10afcd..dc88c1b2dec 100644 --- a/api/src/opentrons/drivers/absorbance_reader/async_byonoy.py +++ b/api/src/opentrons/drivers/absorbance_reader/async_byonoy.py @@ -3,18 +3,20 @@ import re from concurrent.futures.thread import ThreadPoolExecutor from functools import partial -from typing import Optional, List, Dict, Tuple +from typing import Any, Optional, List, Dict, Tuple from .hid_protocol import ( AbsorbanceHidInterface as AbsProtocol, ErrorCodeNames, DeviceStateNames, SlotStateNames, + MeasurementConfig, ) from opentrons.drivers.types import ( AbsorbanceReaderLidStatus, AbsorbanceReaderPlatePresence, AbsorbanceReaderDeviceState, + ABSMeasurementMode, ) from opentrons.drivers.rpi_drivers.types import USBPort from opentrons.hardware_control.modules.errors import AbsorbanceReaderDisconnectedError @@ -22,7 +24,7 @@ SN_PARSER = re.compile(r'ATTRS{serial}=="(?P.+?)"') VERSION_PARSER = re.compile(r"Absorbance (?PV\d+\.\d+\.\d+)") -SERIAL_PARSER = re.compile(r"SN: (?PBYO[A-Z]{3}[0-9]{5})") +SERIAL_PARSER = re.compile(r"(?PBYO[A-Z]{3}[0-9]{5})") class AsyncByonoy: @@ -72,13 +74,13 @@ async def create( loop = loop or asyncio.get_running_loop() executor = ThreadPoolExecutor(max_workers=1) - import pybyonoy_device_library as byonoy # type: ignore[import-not-found] + import byonoy_devices as byonoy # type: ignore[import-not-found] interface: AbsProtocol = byonoy device_sn = cls.serial_number_from_port(usb_port.name) found: List[AbsProtocol.Device] = await loop.run_in_executor( - executor=executor, func=byonoy.byonoy_available_devices + executor=executor, func=byonoy.available_devices ) device = cls.match_device_with_sn(device_sn, found) @@ -110,7 +112,7 @@ def __init__( self._loop = loop self._supported_wavelengths: Optional[list[int]] = None self._device_handle: Optional[int] = None - self._current_config: Optional[AbsProtocol.MeasurementConfig] = None + self._current_config: Optional[MeasurementConfig] = None async def open(self) -> bool: """ @@ -121,7 +123,7 @@ async def open(self) -> bool: err, device_handle = await self._loop.run_in_executor( executor=self._executor, - func=partial(self._interface.byonoy_open_device, self._device), + func=partial(self._interface.open_device, self._device), ) self._raise_if_error(err.name, f"Error opening device: {err}") self._device_handle = device_handle @@ -132,7 +134,7 @@ async def close(self) -> None: handle = self._verify_device_handle() await self._loop.run_in_executor( executor=self._executor, - func=partial(self._interface.byonoy_free_device, handle), + func=partial(self._interface.free_device, handle), ) self._device_handle = None @@ -143,7 +145,7 @@ async def is_open(self) -> bool: handle = self._verify_device_handle() return await self._loop.run_in_executor( executor=self._executor, - func=partial(self._interface.byonoy_device_open, handle), + func=partial(self._interface.device_open, handle), ) async def get_device_information(self) -> Dict[str, str]: @@ -151,7 +153,7 @@ async def get_device_information(self) -> Dict[str, str]: handle = self._verify_device_handle() err, device_info = await self._loop.run_in_executor( executor=self._executor, - func=partial(self._interface.byonoy_get_device_information, handle), + func=partial(self._interface.get_device_information, handle), ) self._raise_if_error(err.name, f"Error getting device information: {err}") serial_match = SERIAL_PARSER.match(device_info.sn) @@ -170,7 +172,7 @@ async def get_device_status(self) -> AbsorbanceReaderDeviceState: handle = self._verify_device_handle() err, status = await self._loop.run_in_executor( executor=self._executor, - func=partial(self._interface.byonoy_get_device_status, handle), + func=partial(self._interface.get_device_status, handle), ) self._raise_if_error(err.name, f"Error getting device status: {err}") return self.convert_device_state(status.name) @@ -182,11 +184,9 @@ async def update_firmware(self, firmware_file_path: str) -> Tuple[bool, str]: return False, f"Firmware file not found: {firmware_file_path}" err = await self._loop.run_in_executor( executor=self._executor, - func=partial( - self._interface.byonoy_update_device, handle, firmware_file_path - ), + func=partial(self._interface.update_device, handle, firmware_file_path), ) - if err.name != "BYONOY_ERROR_NO_ERROR": + if err.name != "NO_ERROR": return False, f"Byonoy update failed with error: {err}" return True, "" @@ -195,7 +195,7 @@ async def get_device_uptime(self) -> int: handle = self._verify_device_handle() err, uptime = await self._loop.run_in_executor( executor=self._executor, - func=partial(self._interface.byonoy_get_device_uptime, handle), + func=partial(self._interface.get_device_uptime, handle), ) self._raise_if_error(err.name, "Error getting device uptime: ") return uptime @@ -205,7 +205,7 @@ async def get_lid_status(self) -> AbsorbanceReaderLidStatus: handle = self._verify_device_handle() err, lid_info = await self._loop.run_in_executor( executor=self._executor, - func=partial(self._interface.byonoy_get_device_parts_aligned, handle), + func=partial(self._interface.get_device_parts_aligned, handle), ) self._raise_if_error(err.name, f"Error getting lid status: {err}") return ( @@ -217,38 +217,38 @@ async def get_supported_wavelengths(self) -> list[int]: handle = self._verify_device_handle() err, wavelengths = await self._loop.run_in_executor( executor=self._executor, - func=partial( - self._interface.byonoy_abs96_get_available_wavelengths, handle - ), + func=partial(self._interface.abs96_get_available_wavelengths, handle), ) self._raise_if_error(err.name, "Error getting available wavelengths: ") self._supported_wavelengths = wavelengths return wavelengths - async def get_single_measurement(self, wavelength: int) -> List[float]: - """Get a single measurement based on the current configuration.""" + async def get_measurement(self) -> List[List[float]]: + """Gets one or more measurements based on the current configuration.""" handle = self._verify_device_handle() assert ( - self._current_config - and self._current_config.sample_wavelength == wavelength - ) + self._current_config is not None + ), "Cannot get measurement without initializing." + measure_func: Any = self._interface.abs96_single_measure + if isinstance(self._current_config, AbsProtocol.MultiMeasurementConfig): + measure_func = self._interface.abs96_multiple_measure err, measurements = await self._loop.run_in_executor( executor=self._executor, func=partial( - self._interface.byonoy_abs96_single_measure, + measure_func, handle, self._current_config, ), ) - self._raise_if_error(err.name, f"Error getting single measurement: {err}") - return measurements + self._raise_if_error(err.name, f"Error getting measurement: {err}") + return measurements if isinstance(measurements[0], List) else [measurements] # type: ignore async def get_plate_presence(self) -> AbsorbanceReaderPlatePresence: """Get the state of the plate for the reader.""" handle = self._verify_device_handle() err, presence = await self._loop.run_in_executor( executor=self._executor, - func=partial(self._interface.byonoy_get_device_slot_status, handle), + func=partial(self._interface.get_device_slot_status, handle), ) self._raise_if_error(err.name, f"Error getting slot status: {err}") return self.convert_plate_presence(presence.name) @@ -256,40 +256,54 @@ async def get_plate_presence(self) -> AbsorbanceReaderPlatePresence: def _get_supported_wavelengths(self) -> List[int]: handle = self._verify_device_handle() wavelengths: List[int] - err, wavelengths = self._interface.byonoy_abs96_get_available_wavelengths( - handle - ) + err, wavelengths = self._interface.abs96_get_available_wavelengths(handle) self._raise_if_error(err.name, f"Error getting available wavelengths: {err}") self._supported_wavelengths = wavelengths return wavelengths - def _initialize_measurement(self, conf: AbsProtocol.MeasurementConfig) -> None: + def _initialize_measurement(self, conf: MeasurementConfig) -> None: handle = self._verify_device_handle() - err = self._interface.byonoy_abs96_initialize_single_measurement(handle, conf) + if isinstance(conf, AbsProtocol.SingleMeasurementConfig): + err = self._interface.abs96_initialize_single_measurement(handle, conf) + else: + err = self._interface.abs96_initialize_multiple_measurement(handle, conf) self._raise_if_error(err.name, f"Error initializing measurement: {err}") self._current_config = conf - def _set_sample_wavelength(self, wavelength: int) -> AbsProtocol.MeasurementConfig: + def _initialize( + self, + mode: ABSMeasurementMode, + wavelengths: List[int], + reference_wavelength: Optional[int] = None, + ) -> None: if not self._supported_wavelengths: self._get_supported_wavelengths() assert self._supported_wavelengths - if wavelength in self._supported_wavelengths: - conf = self._interface.ByonoyAbs96SingleMeasurementConfig() - conf.sample_wavelength = wavelength - return conf + conf: MeasurementConfig + if set(wavelengths).issubset(self._supported_wavelengths): + if mode == ABSMeasurementMode.SINGLE: + conf = self._interface.Abs96SingleMeasurementConfig() + conf.sample_wavelength = wavelengths[0] or 0 + conf.reference_wavelength = reference_wavelength or 0 + else: + conf = self._interface.Abs96MultipleMeasurementConfig() + conf.sample_wavelengths = wavelengths else: raise ValueError( - f"Unsupported wavelength: {wavelength}, expected: {self._supported_wavelengths}" + f"Unsupported wavelength: {wavelengths}, expected: {self._supported_wavelengths}" ) - - def _initialize(self, wavelength: int) -> None: - conf = self._set_sample_wavelength(wavelength) self._initialize_measurement(conf) - async def initialize(self, wavelength: int) -> None: - """Initialize the device so we can start reading samples from it.""" + async def initialize( + self, + mode: ABSMeasurementMode, + wavelengths: List[int], + reference_wavelength: Optional[int] = None, + ) -> None: + """initialize the device so we can start reading samples from it.""" await self._loop.run_in_executor( - executor=self._executor, func=partial(self._initialize, wavelength) + executor=self._executor, + func=partial(self._initialize, mode, wavelengths, reference_wavelength), ) def _verify_device_handle(self) -> int: @@ -304,12 +318,12 @@ def _raise_if_error( msg: str = "Error occurred: ", ) -> None: if err_name in [ - "BYONOY_ERROR_DEVICE_CLOSED", - "BYONOY_ERROR_DEVICE_COMMUNICATION_FAILURE", - "BYONOY_ERROR_UNSUPPORTED_OPERATION", + "DEVICE_CLOSED", + "DEVICE_COMMUNICATION_FAILURE", + "UNSUPPORTED_OPERATION", ]: raise AbsorbanceReaderDisconnectedError(self._device.sn) - if err_name != "BYONOY_ERROR_NO_ERROR": + if err_name != "NO_ERROR": raise RuntimeError(msg, err_name) @staticmethod diff --git a/api/src/opentrons/drivers/absorbance_reader/driver.py b/api/src/opentrons/drivers/absorbance_reader/driver.py index be6fdaa5c15..5899fef89d0 100644 --- a/api/src/opentrons/drivers/absorbance_reader/driver.py +++ b/api/src/opentrons/drivers/absorbance_reader/driver.py @@ -4,11 +4,14 @@ from typing import Dict, Optional, List, Tuple, TYPE_CHECKING from opentrons.drivers.types import ( + ABSMeasurementMode, AbsorbanceReaderLidStatus, AbsorbanceReaderDeviceState, AbsorbanceReaderPlatePresence, ) -from opentrons.drivers.absorbance_reader.abstract import AbstractAbsorbanceReaderDriver +from opentrons.drivers.absorbance_reader.abstract import ( + AbstractAbsorbanceReaderDriver, +) from opentrons.drivers.rpi_drivers.types import USBPort if TYPE_CHECKING: @@ -60,12 +63,16 @@ async def get_lid_status(self) -> AbsorbanceReaderLidStatus: async def get_available_wavelengths(self) -> List[int]: return await self._connection.get_supported_wavelengths() - async def get_single_measurement(self, wavelength: int) -> List[float]: - # TODO (cb, 08-02-2024): The list of measurements for 96 wells is rotated 180 degrees (well A1 is where well H12 should be) this must be corrected - return await self._connection.get_single_measurement(wavelength) + async def initialize_measurement( + self, + wavelengths: List[int], + mode: ABSMeasurementMode = ABSMeasurementMode.SINGLE, + reference_wavelength: Optional[int] = None, + ) -> None: + await self._connection.initialize(mode, wavelengths, reference_wavelength) - async def initialize_measurement(self, wavelength: int) -> None: - await self._connection.initialize(wavelength) + async def get_measurement(self) -> List[List[float]]: + return await self._connection.get_measurement() async def get_status(self) -> AbsorbanceReaderDeviceState: return await self._connection.get_device_status() diff --git a/api/src/opentrons/drivers/absorbance_reader/hid_protocol.py b/api/src/opentrons/drivers/absorbance_reader/hid_protocol.py index 8816b2e2903..c418a652130 100644 --- a/api/src/opentrons/drivers/absorbance_reader/hid_protocol.py +++ b/api/src/opentrons/drivers/absorbance_reader/hid_protocol.py @@ -1,9 +1,11 @@ from typing import ( Dict, + Optional, Protocol, List, Literal, Tuple, + Union, runtime_checkable, TypeVar, ) @@ -11,28 +13,28 @@ Response = TypeVar("Response") ErrorCodeNames = Literal[ - "BYONOY_ERROR_NO_ERROR", - "BYONOY_ERROR_UNKNOWN_ERROR", - "BYONOY_ERROR_DEVICE_CLOSED", - "BYONOY_ERROR_INVALID_ARGUMENT", - "BYONOY_ERROR_NO_MEMORY", - "BYONOY_ERROR_UNSUPPORTED_OPERATION", - "BYONOY_ERROR_DEVICE_COMMUNICATION_FAILURE", - "BYONOY_ERROR_DEVICE_OPERATION_FAILED", - "BYONOY_ERROR_DEVICE_OPEN_PREFIX", - "BYONOY_ERROR_DEVICE_NOT_FOUND", - "BYONOY_ERROR_DEVICE_TOO_NEW", - "BYONOY_ERROR_DEVICE_ALREADY_OPEN", - "BYONOY_ERROR_FIRMWARE_UPDATE_ERROR_PREFIX", - "BYONOY_ERROR_FIRMWARE_UPDATE_FILE_NOT_FOUND", - "BYONOY_ERROR_FIRMWARE_UPDATE_FILE_NOT_VALID", - "BYONOY_ERROR_FIRMWARE_UPDATE_FAILED", - "BYONOY_ERROR_FILE_ERROR_PREFIX", - "BYONOY_ERROR_FILE_WRITE_ERROR", - "BYONOY_ERROR_MEASUTEMNT_ERROR_PREFIX", - "BYONOY_ERROR_MEASUTEMNT_SLOT_NOT_EMPTY", - "BYONOY_ERROR_NOT_INITIALIZED", - "BYONOY_ERROR_INTERNAL", + "NO_ERROR", + "UNKNOWN_ERROR", + "DEVICE_CLOSED", + "INVALID_ARGUMENT", + "NO_MEMORY", + "UNSUPPORTED_OPERATION", + "DEVICE_COMMUNICATION_FAILURE", + "DEVICE_OPERATION_FAILED", + "DEVICE_OPEN_PREFIX", + "DEVICE_NOT_FOUND", + "DEVICE_TOO_NEW", + "DEVICE_ALREADY_OPEN", + "FIRMWARE_UPDATE_ERROR_PREFIX", + "FIRMWARE_UPDATE_FILE_NOT_FOUND", + "FIRMWARE_UPDATE_FILE_NOT_VALID", + "FIRMWARE_UPDATE_FAILED", + "FILE_ERROR_PREFIX", + "FILE_WRITE_ERROR", + "MEASUTEMNT_ERROR_PREFIX", + "MEASUTEMNT_SLOT_NOT_EMPTY", + "NOT_INITIALIZED", + "INTERNAL", ] SlotStateNames = Literal[ @@ -69,8 +71,13 @@ class SlotState(Protocol): value: int @runtime_checkable - class MeasurementConfig(Protocol): + class SingleMeasurementConfig(Protocol): sample_wavelength: int + reference_wavelength: Optional[int] + + @runtime_checkable + class MultiMeasurementConfig(Protocol): + sample_wavelengths: List[int] @runtime_checkable class DeviceInfo(Protocol): @@ -84,60 +91,71 @@ class DeviceState(Protocol): name: DeviceStateNames value: int - def ByonoyAbs96SingleMeasurementConfig(self) -> MeasurementConfig: + def Abs96SingleMeasurementConfig(self) -> SingleMeasurementConfig: + ... + + def Abs96MultipleMeasurementConfig(self) -> MultiMeasurementConfig: ... - def byonoy_open_device(self, device: Device) -> Tuple[ErrorCode, int]: + def open_device(self, device: Device) -> Tuple[ErrorCode, int]: ... - def byonoy_free_device(self, device_handle: int) -> Tuple[ErrorCode, bool]: + def free_device(self, device_handle: int) -> Tuple[ErrorCode, bool]: ... - def byonoy_device_open(self, device_handle: int) -> bool: + def device_open(self, device_handle: int) -> bool: ... - def byonoy_get_device_information( + def get_device_information( self, device_handle: int ) -> Tuple[ErrorCode, DeviceInfo]: ... - def byonoy_update_device( - self, device_handle: int, firmware_file_path: str - ) -> ErrorCode: + def update_device(self, device_handle: int, firmware_file_path: str) -> ErrorCode: ... - def byonoy_get_device_status( - self, device_handle: int - ) -> Tuple[ErrorCode, DeviceState]: + def get_device_status(self, device_handle: int) -> Tuple[ErrorCode, DeviceState]: ... - def byonoy_get_device_uptime(self, device_handle: int) -> Tuple[ErrorCode, int]: + def get_device_uptime(self, device_handle: int) -> Tuple[ErrorCode, int]: ... - def byonoy_get_device_slot_status( - self, device_handle: int - ) -> Tuple[ErrorCode, SlotState]: + def get_device_slot_status(self, device_handle: int) -> Tuple[ErrorCode, SlotState]: ... - def byonoy_get_device_parts_aligned( - self, device_handle: int - ) -> Tuple[ErrorCode, bool]: + def get_device_parts_aligned(self, device_handle: int) -> Tuple[ErrorCode, bool]: ... - def byonoy_abs96_get_available_wavelengths( + def abs96_get_available_wavelengths( self, device_handle: int ) -> Tuple[ErrorCode, List[int]]: ... - def byonoy_abs96_initialize_single_measurement( - self, device_handle: int, conf: MeasurementConfig + def abs96_initialize_single_measurement( + self, device_handle: int, conf: SingleMeasurementConfig ) -> ErrorCode: ... - def byonoy_abs96_single_measure( - self, device_handle: int, conf: MeasurementConfig + def abs96_initialize_multiple_measurement( + self, device_handle: int, conf: MultiMeasurementConfig + ) -> ErrorCode: + ... + + def abs96_single_measure( + self, device_handle: int, conf: SingleMeasurementConfig ) -> Tuple[ErrorCode, List[float]]: ... - def byonoy_available_devices(self) -> List[Device]: + def abs96_multiple_measure( + self, device_handle: int, conf: MultiMeasurementConfig + ) -> Tuple[ErrorCode, List[List[float]]]: + ... + + def available_devices(self) -> List[Device]: ... + + +MeasurementConfig = Union[ + AbsorbanceHidInterface.SingleMeasurementConfig, + AbsorbanceHidInterface.MultiMeasurementConfig, +] diff --git a/api/src/opentrons/drivers/absorbance_reader/simulator.py b/api/src/opentrons/drivers/absorbance_reader/simulator.py index 6a2a0aa6228..c4d7af2c1eb 100644 --- a/api/src/opentrons/drivers/absorbance_reader/simulator.py +++ b/api/src/opentrons/drivers/absorbance_reader/simulator.py @@ -2,6 +2,7 @@ from opentrons.util.async_helpers import ensure_yield from opentrons.drivers.types import ( + ABSMeasurementMode, AbsorbanceReaderLidStatus, AbsorbanceReaderDeviceState, AbsorbanceReaderPlatePresence, @@ -54,11 +55,16 @@ async def get_available_wavelengths(self) -> List[int]: return [450, 570, 600, 650] @ensure_yield - async def get_single_measurement(self, wavelength: int) -> List[float]: - return [0.0] + async def get_measurement(self) -> List[List[float]]: + return [[0.0]] @ensure_yield - async def initialize_measurement(self, wavelength: int) -> None: + async def initialize_measurement( + self, + wavelengths: List[int], + mode: ABSMeasurementMode = ABSMeasurementMode.SINGLE, + reference_wavelength: Optional[int] = None, + ) -> None: pass @ensure_yield diff --git a/api/src/opentrons/drivers/types.py b/api/src/opentrons/drivers/types.py index a4047ca64da..f01001bdef8 100644 --- a/api/src/opentrons/drivers/types.py +++ b/api/src/opentrons/drivers/types.py @@ -1,6 +1,6 @@ """ Type definitions for modules in this tree """ from dataclasses import dataclass -from typing import Dict, NamedTuple, Optional +from typing import Any, Dict, List, NamedTuple, Optional from enum import Enum @@ -83,3 +83,25 @@ class AbsorbanceReaderDeviceState(str, Enum): OK = "ok" BROKEN_FW = "broken_fw" ERROR = "error" + + +class ABSMeasurementMode(Enum): + """The current mode configured for reading the Absorbance Reader.""" + + SINGLE = "single" + MULTI = "multi" + + +@dataclass +class ABSMeasurementConfig: + measure_mode: ABSMeasurementMode + sample_wavelengths: List[int] + reference_wavelength: Optional[int] + + @property + def data(self) -> Dict[str, Any]: + return { + "measureMode": self.measure_mode.value, + "sampleWavelengths": self.sample_wavelengths, + "referenceWavelength": self.reference_wavelength, + } diff --git a/api/src/opentrons/hardware_control/modules/absorbance_reader.py b/api/src/opentrons/hardware_control/modules/absorbance_reader.py index e68b535c4bc..da7c4746086 100644 --- a/api/src/opentrons/hardware_control/modules/absorbance_reader.py +++ b/api/src/opentrons/hardware_control/modules/absorbance_reader.py @@ -12,6 +12,8 @@ AbsorbanceReaderLidStatus, AbsorbanceReaderPlatePresence, AbsorbanceReaderDeviceState, + ABSMeasurementMode, + ABSMeasurementConfig, ) from opentrons.hardware_control.execution_manager import ExecutionManager @@ -194,13 +196,21 @@ def __init__( self._device_info = device_info self._reader = reader self._poller = poller + self._measurement_config: Optional[ABSMeasurementConfig] = None + self._device_status = AbsorbanceReaderStatus.IDLE self._error: Optional[str] = None self._reader.register_error_handler(self._enter_error_state) @property def status(self) -> AbsorbanceReaderStatus: - """Return some string describing status.""" - return AbsorbanceReaderStatus.IDLE + """Return some string describing the device status.""" + state = self._reader.device_state + if state not in [ + AbsorbanceReaderDeviceState.UNKNOWN, + AbsorbanceReaderDeviceState.OK, + ]: + return AbsorbanceReaderStatus.ERROR + return self._device_status @property def lid_status(self) -> AbsorbanceReaderLidStatus: @@ -210,6 +220,20 @@ def lid_status(self) -> AbsorbanceReaderLidStatus: def plate_presence(self) -> AbsorbanceReaderPlatePresence: return self._reader.plate_presence + @property + def uptime(self) -> int: + """Time in ms this device has been running for.""" + return self._reader.uptime + + @property + def supported_wavelengths(self) -> List[int]: + """The wavelengths in nm this plate reader supports.""" + return self._reader.supported_wavelengths + + @property + def measurement_config(self) -> Optional[ABSMeasurementConfig]: + return self._measurement_config + @property def device_info(self) -> Mapping[str, str]: """Return a dict of the module's static information (serial, etc)""" @@ -218,12 +242,17 @@ def device_info(self) -> Mapping[str, str]: @property def live_data(self) -> LiveData: """Return a dict of the module's dynamic information""" + conf = self._measurement_config.data if self._measurement_config else dict() return { "status": self.status.value, "data": { + "uptime": self.uptime, + "deviceStatus": self.status.value, "lidStatus": self.lid_status.value, "platePresence": self.plate_presence.value, - "sampleWavelength": 400, + "measureMode": conf.get("measureMode", ""), + "sampleWavelengths": conf.get("sampleWavelengths", []), + "referenceWavelength": conf.get("referenceWavelength", 0), }, } @@ -309,17 +338,32 @@ async def cleanup(self) -> None: """ await self.deactivate() - async def set_sample_wavelength(self, wavelength: int) -> None: - """Set the Absorbance Reader's active wavelength.""" - await self._driver.initialize_measurement(wavelength) - - async def start_measure(self, wavelength: int) -> List[float]: - """Initiate a single measurement.""" - return await self._driver.get_single_measurement(wavelength) + async def set_sample_wavelength( + self, + mode: ABSMeasurementMode, + wavelengths: List[int], + reference_wavelength: Optional[int] = None, + ) -> None: + """Set the Absorbance Reader's measurement mode and active wavelength.""" + if mode == ABSMeasurementMode.SINGLE: + assert ( + len(wavelengths) == 1 + ), "Cannot initialize single read mode with more than 1 wavelength." + + await self._driver.initialize_measurement(wavelengths, mode) + self._measurement_config = ABSMeasurementConfig( + measure_mode=mode, + sample_wavelengths=wavelengths, + reference_wavelength=reference_wavelength, + ) - async def get_current_wavelength(self) -> None: - """Get the Absorbance Reader's current active wavelength.""" - pass # TODO: implement + async def start_measure(self) -> List[List[float]]: + """Initiate a measurement depending on the measurement mode.""" + try: + self._device_status = AbsorbanceReaderStatus.MEASURING + return await self._driver.get_measurement() + finally: + self._device_status = AbsorbanceReaderStatus.IDLE async def get_current_lid_status(self) -> AbsorbanceReaderLidStatus: """Get the Absorbance Reader's current lid status.""" diff --git a/api/src/opentrons/hardware_control/modules/types.py b/api/src/opentrons/hardware_control/modules/types.py index c7b3c920c79..6c9f6a3e915 100644 --- a/api/src/opentrons/hardware_control/modules/types.py +++ b/api/src/opentrons/hardware_control/modules/types.py @@ -3,6 +3,7 @@ from dataclasses import dataclass from typing import ( Dict, + List, NamedTuple, Callable, Any, @@ -46,7 +47,7 @@ class ThermocyclerStep(ThermocyclerStepBase, total=False): class LiveData(TypedDict): status: str - data: Dict[str, Union[float, str, bool, None]] + data: Dict[str, Union[float, str, bool, List[int], None]] class ModuleType(str, Enum): diff --git a/api/src/opentrons/hardware_control/ot3api.py b/api/src/opentrons/hardware_control/ot3api.py index bea325cfb83..e43eb674e12 100644 --- a/api/src/opentrons/hardware_control/ot3api.py +++ b/api/src/opentrons/hardware_control/ot3api.py @@ -164,6 +164,7 @@ def _adjust_high_throughput_z_current(func: Wrapped) -> Wrapped: A decorator that temproarily and conditionally changes the active current (based on the axis input) before a function is executed and the cleans up afterwards """ + # only home and retract should be wrappeed by this decorator @wraps(func) async def wrapper(self: Any, axis: Axis, *args: Any, **kwargs: Any) -> Any: @@ -933,7 +934,6 @@ async def home_gear_motors(self) -> None: current_pos_float > self._config.safe_home_distance and current_pos_float < max_distance ): - # move toward home until a safe distance await self._backend.tip_action( origin={Axis.Q: current_pos_float}, @@ -1811,7 +1811,8 @@ async def tip_pickup_moves( increment: Optional[float] = None, ) -> None: """This is a slightly more barebones variation of pick_up_tip. This is only the motor routine - directly involved in tip pickup, and leaves any state updates and plunger moves to the caller.""" + directly involved in tip pickup, and leaves any state updates and plunger moves to the caller. + """ realmount = OT3Mount.from_mount(mount) instrument = self._pipette_handler.get_pipette(realmount) @@ -2654,7 +2655,7 @@ async def _liquid_probe_pass( cp = self.critical_point_for(mount, None) return deck_end_z + offset.z + cp.z - async def liquid_probe( + async def liquid_probe( # noqa: C901 self, mount: Union[top_types.Mount, OT3Mount], max_z_dist: float, @@ -2683,6 +2684,16 @@ async def liquid_probe( self._pipette_handler.ready_for_tip_action( instrument, HardwareAction.LIQUID_PROBE, checked_mount ) + # default to using all available sensors + if probe: + checked_probe = probe + else: + checked_probe = ( + InstrumentProbeType.BOTH + if instrument.channels > 1 + else InstrumentProbeType.PRIMARY + ) + if not probe_settings: probe_settings = deepcopy(self.config.liquid_sense) @@ -2774,7 +2785,7 @@ async def prep_plunger_for_probe_move( height = await self._liquid_probe_pass( checked_mount, probe_settings, - probe if probe else InstrumentProbeType.PRIMARY, + checked_probe, plunger_travel_mm + sensor_baseline_plunger_move_mm, ) # if we made it here without an error we found the liquid diff --git a/api/src/opentrons/protocol_api/core/engine/deck_conflict.py b/api/src/opentrons/protocol_api/core/engine/deck_conflict.py index 6ebb47f0ac8..abf47212dac 100644 --- a/api/src/opentrons/protocol_api/core/engine/deck_conflict.py +++ b/api/src/opentrons/protocol_api/core/engine/deck_conflict.py @@ -16,6 +16,7 @@ from opentrons_shared_data.errors.exceptions import MotionPlanningFailureError from opentrons_shared_data.module import FLEX_TC_LID_COLLISION_ZONE +from opentrons.hardware_control import CriticalPoint from opentrons.hardware_control.modules.types import ModuleType from opentrons.motion_planning import deck_conflict as wrapped_deck_conflict from opentrons.motion_planning import adjacent_slots_getters @@ -228,9 +229,13 @@ def check_safe_for_pipette_movement( ) primary_nozzle = engine_state.pipettes.get_primary_nozzle(pipette_id) + destination_cp = _get_critical_point_to_use(engine_state, labware_id) + pipette_bounds_at_well_location = ( engine_state.pipettes.get_pipette_bounds_at_specified_move_to_position( - pipette_id=pipette_id, destination_position=well_location_point + pipette_id=pipette_id, + destination_position=well_location_point, + critical_point=destination_cp, ) ) if not _is_within_pipette_extents( @@ -284,6 +289,21 @@ def check_safe_for_pipette_movement( ) +def _get_critical_point_to_use( + engine_state: StateView, labware_id: str +) -> Optional[CriticalPoint]: + """Return the critical point to use when accessing the given labware.""" + # TODO (spp, 2024-09-17): looks like Y_CENTER of column is the same as its XY_CENTER. + # I'm using this if-else ladder to be consistent with what we do in + # `MotionPlanning.get_movement_waypoints_to_well()`. + # We should probably use only XY_CENTER in both places. + if engine_state.labware.get_should_center_column_on_target_well(labware_id): + return CriticalPoint.Y_CENTER + elif engine_state.labware.get_should_center_pipette_on_target_well(labware_id): + return CriticalPoint.XY_CENTER + return None + + def _slot_has_potential_colliding_object( engine_state: StateView, pipette_bounds: Tuple[Point, Point, Point, Point], diff --git a/api/src/opentrons/protocol_api/core/engine/module_core.py b/api/src/opentrons/protocol_api/core/engine/module_core.py index 643111122ce..729037425a8 100644 --- a/api/src/opentrons/protocol_api/core/engine/module_core.py +++ b/api/src/opentrons/protocol_api/core/engine/module_core.py @@ -18,6 +18,7 @@ ) from opentrons.protocol_engine import commands as cmd +from opentrons.protocol_engine.types import ABSMeasureMode from opentrons.types import DeckSlotName from opentrons.protocol_engine.clients import SyncClient as ProtocolEngineClient from opentrons.protocol_engine.errors.exceptions import ( @@ -525,25 +526,31 @@ class AbsorbanceReaderCore(ModuleCore, AbstractAbsorbanceReaderCore): """Absorbance Reader core logic implementation for Python protocols.""" _sync_module_hardware: SynchronousAdapter[hw_modules.AbsorbanceReader] - _initialized_value: Optional[int] = None + _initialized_value: Optional[List[int]] = None - def initialize(self, wavelength: int) -> None: + def initialize( + self, + mode: ABSMeasureMode, + wavelengths: List[int], + reference_wavelength: Optional[int] = None, + ) -> None: """Initialize the Absorbance Reader by taking zero reading.""" + # TODO: check that the wavelengths are within the supported wavelengths self._engine_client.execute_command( cmd.absorbance_reader.InitializeParams( moduleId=self.module_id, - sampleWavelength=wavelength, + measureMode=mode, + sampleWavelengths=wavelengths, + referenceWavelength=reference_wavelength, ), ) - self._initialized_value = wavelength + self._initialized_value = wavelengths - def read(self) -> Optional[Dict[str, float]]: + def read(self) -> Optional[Dict[int, Dict[str, float]]]: """Initiate a read on the Absorbance Reader, and return the results. During Analysis, this will return None.""" if self._initialized_value: self._engine_client.execute_command( - cmd.absorbance_reader.ReadAbsorbanceParams( - moduleId=self.module_id, sampleWavelength=self._initialized_value - ) + cmd.absorbance_reader.ReadAbsorbanceParams(moduleId=self.module_id) ) if not self._engine_client.state.config.use_virtual_modules: read_result = ( diff --git a/api/src/opentrons/protocol_api/core/module.py b/api/src/opentrons/protocol_api/core/module.py index e6968de91d6..90abea1d0ec 100644 --- a/api/src/opentrons/protocol_api/core/module.py +++ b/api/src/opentrons/protocol_api/core/module.py @@ -16,6 +16,7 @@ MagneticStatus, SpeedStatus, ) +from opentrons.protocol_engine.types import ABSMeasureMode from opentrons.types import DeckSlotName @@ -355,11 +356,16 @@ def get_serial_number(self) -> str: """Get the module's unique hardware serial number.""" @abstractmethod - def initialize(self, wavelength: int) -> None: + def initialize( + self, + mode: ABSMeasureMode, + wavelengths: List[int], + reference_wavelength: Optional[int] = None, + ) -> None: """Initialize the Absorbance Reader by taking zero reading.""" @abstractmethod - def read(self) -> Optional[Dict[str, float]]: + def read(self) -> Optional[Dict[int, Dict[str, float]]]: """Get an absorbance reading from the Absorbance Reader.""" @abstractmethod diff --git a/api/src/opentrons/protocol_api/instrument_context.py b/api/src/opentrons/protocol_api/instrument_context.py index 7121567c3c4..e11ffcc78c2 100644 --- a/api/src/opentrons/protocol_api/instrument_context.py +++ b/api/src/opentrons/protocol_api/instrument_context.py @@ -261,6 +261,7 @@ def aspirate( and well is not None and self.liquid_presence_detection and self._96_tip_config_valid() + and self._core.get_current_volume() == 0 ): self.require_liquid_presence(well=well) diff --git a/api/src/opentrons/protocol_api/module_contexts.py b/api/src/opentrons/protocol_api/module_contexts.py index b2abeca24e4..f9fcc18ca00 100644 --- a/api/src/opentrons/protocol_api/module_contexts.py +++ b/api/src/opentrons/protocol_api/module_contexts.py @@ -3,6 +3,7 @@ import logging from typing import List, Dict, Optional, Union, cast +from opentrons.protocol_engine.types import ABSMeasureMode from opentrons_shared_data.labware.types import LabwareDefinition from opentrons_shared_data.module.types import ModuleModel, ModuleType @@ -1003,11 +1004,31 @@ def is_lid_on(self) -> bool: return self._core.is_lid_on() @requires_version(2, 21) - def initialize(self, wavelength: int) -> None: - """Initialize the Absorbance Reader by taking zero reading.""" - self._core.initialize(wavelength) + def initialize( + self, + mode: ABSMeasureMode, + wavelengths: List[int], + reference_wavelength: Optional[int] = None, + ) -> None: + """Take a zero reading on the Absorbance Plate Reader Module. + + :param mode: Either ``"single"`` or ``"multi"``. + + - In single measurement mode, :py:meth:`.AbsorbanceReaderContext.read` uses + one sample wavelength and an optional reference wavelength. + - In multiple measurement mode, :py:meth:`.AbsorbanceReaderContext.read` uses + a list of up to six sample wavelengths. + :param wavelengths: A list of wavelengths, in mm, to measure. + - Must contain only one item when initializing a single measurement. + - Must contain one to six items when initializing a multiple measurement. + :param reference_wavelength: An optional reference wavelength, in mm. Cannot be + used with multiple measurements. + """ + self._core.initialize( + mode, wavelengths, reference_wavelength=reference_wavelength + ) @requires_version(2, 21) - def read(self) -> Optional[Dict[str, float]]: - """Initiate read on the Absorbance Reader. Returns a dictionary of values ordered by well name.""" + def read(self) -> Optional[Dict[int, Dict[str, float]]]: + """Initiate read on the Absorbance Reader. Returns a dictionary of wavelengths to dictionary of values ordered by well name.""" return self._core.read() diff --git a/api/src/opentrons/protocol_engine/commands/absorbance_reader/close_lid.py b/api/src/opentrons/protocol_engine/commands/absorbance_reader/close_lid.py index 4b1135668d0..f7d7b999104 100644 --- a/api/src/opentrons/protocol_engine/commands/absorbance_reader/close_lid.py +++ b/api/src/opentrons/protocol_engine/commands/absorbance_reader/close_lid.py @@ -12,6 +12,8 @@ from opentrons.protocol_engine.resources import labware_validation from .types import MoveLidResult +from ...state.update_types import StateUpdate + from opentrons.drivers.types import AbsorbanceReaderLidStatus @@ -132,11 +134,19 @@ async def execute( labware_location=new_location, ) + state_update = StateUpdate() + state_update.set_labware_location( + labware_id=loaded_lid.id, + new_location=new_location, + new_offset_id=new_offset_id, + ) + return SuccessData( public=CloseLidResult( lidId=loaded_lid.id, newLocation=new_location, offsetId=new_offset_id ), private=None, + state_update=state_update, ) diff --git a/api/src/opentrons/protocol_engine/commands/absorbance_reader/initialize.py b/api/src/opentrons/protocol_engine/commands/absorbance_reader/initialize.py index 53150280a80..314645b39b2 100644 --- a/api/src/opentrons/protocol_engine/commands/absorbance_reader/initialize.py +++ b/api/src/opentrons/protocol_engine/commands/absorbance_reader/initialize.py @@ -1,10 +1,13 @@ """Command models to initialize an Absorbance Reader.""" from __future__ import annotations -from typing import Optional, Literal, TYPE_CHECKING +from typing import List, Optional, Literal, TYPE_CHECKING from typing_extensions import Type from pydantic import BaseModel, Field +from opentrons.drivers.types import ABSMeasurementMode +from opentrons.protocol_engine.types import ABSMeasureMode + from ..command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData from ...errors.error_occurrence import ErrorOccurrence @@ -20,7 +23,13 @@ class InitializeParams(BaseModel): """Input parameters to initialize an absorbance reading.""" moduleId: str = Field(..., description="Unique ID of the absorbance reader.") - sampleWavelength: int = Field(..., description="Sample wavelength in nm.") + measureMode: ABSMeasureMode = Field( + ..., description="Initialize single or multi measurement mode." + ) + sampleWavelengths: List[int] = Field(..., description="Sample wavelengths in nm.") + referenceWavelength: Optional[int] = Field( + None, description="Optional reference wavelength in nm." + ) class InitializeResult(BaseModel): @@ -54,7 +63,45 @@ async def execute( ) if abs_reader is not None: - await abs_reader.set_sample_wavelength(wavelength=params.sampleWavelength) + # Validate the parameters before initializing. + sample_wavelengths = set(params.sampleWavelengths) + sample_wavelengths_len = len(params.sampleWavelengths) + reference_wavelength = params.referenceWavelength + supported_wavelengths = set(abs_reader.supported_wavelengths) + unsupported_wavelengths = sample_wavelengths.difference( + supported_wavelengths + ) + if unsupported_wavelengths: + raise ValueError(f"Unsupported wavelengths: {unsupported_wavelengths}") + + if params.measureMode == "single": + if sample_wavelengths_len != 1: + raise ValueError( + f"single requires one sample wavelength, provided {sample_wavelengths}" + ) + if ( + reference_wavelength is not None + and reference_wavelength not in supported_wavelengths + ): + raise ValueError( + f"Reference wavelength {reference_wavelength} not supported {supported_wavelengths}" + ) + + if params.measureMode == "multi": + if sample_wavelengths_len < 1 or sample_wavelengths_len > 6: + raise ValueError( + f"multi requires 1-6 sample wavelengths, provided {sample_wavelengths}" + ) + if reference_wavelength is not None: + raise RuntimeError( + "Reference wavelength cannot be used with multi mode." + ) + + await abs_reader.set_sample_wavelength( + ABSMeasurementMode(params.measureMode), + params.sampleWavelengths, + reference_wavelength=params.referenceWavelength, + ) return SuccessData( public=InitializeResult(), diff --git a/api/src/opentrons/protocol_engine/commands/absorbance_reader/open_lid.py b/api/src/opentrons/protocol_engine/commands/absorbance_reader/open_lid.py index e6da9edade5..39fa4ce6ad4 100644 --- a/api/src/opentrons/protocol_engine/commands/absorbance_reader/open_lid.py +++ b/api/src/opentrons/protocol_engine/commands/absorbance_reader/open_lid.py @@ -15,6 +15,9 @@ from opentrons.drivers.types import AbsorbanceReaderLidStatus +from ...state.update_types import StateUpdate + + if TYPE_CHECKING: from opentrons.protocol_engine.state.state import StateView from opentrons.protocol_engine.execution import ( @@ -125,6 +128,14 @@ async def execute(self, params: OpenLidParams) -> SuccessData[OpenLidResult, Non labware_location=new_location, ) + state_update = StateUpdate() + + state_update.set_labware_location( + labware_id=loaded_lid.id, + new_location=new_location, + new_offset_id=new_offset_id, + ) + return SuccessData( public=OpenLidResult( lidId=loaded_lid.id, @@ -132,6 +143,7 @@ async def execute(self, params: OpenLidParams) -> SuccessData[OpenLidResult, Non offsetId=new_offset_id, ), private=None, + state_update=state_update, ) diff --git a/api/src/opentrons/protocol_engine/commands/absorbance_reader/read.py b/api/src/opentrons/protocol_engine/commands/absorbance_reader/read.py index f361f819c5e..b101cdb70b8 100644 --- a/api/src/opentrons/protocol_engine/commands/absorbance_reader/read.py +++ b/api/src/opentrons/protocol_engine/commands/absorbance_reader/read.py @@ -18,16 +18,17 @@ class ReadAbsorbanceParams(BaseModel): - """Input parameters for a single absorbance reading.""" + """Input parameters for an absorbance reading.""" moduleId: str = Field(..., description="Unique ID of the Absorbance Reader.") - sampleWavelength: int = Field(..., description="Sample wavelength in nm.") class ReadAbsorbanceResult(BaseModel): - """Result data from running an aborbance reading, returned as a dictionary map of values by well name (eg. ("A1": 0.0, ...)).""" + """Result data from running an aborbance reading, returned as a dictionary map of wavelengths containing a map of values by well name (eg. {450: {"A1": 0.0, ...}}).""" - data: Optional[Dict[str, float]] = Field(..., description="Absorbance data points.") + data: Optional[Dict[int, Dict[str, float]]] = Field( + ..., description="Absorbance data points per wavelength." + ) class ReadAbsorbanceImpl( @@ -47,7 +48,7 @@ def __init__( async def execute( self, params: ReadAbsorbanceParams ) -> SuccessData[ReadAbsorbanceResult, None]: - """Initiate a single absorbance measurement.""" + """Initiate an absorbance measurement.""" abs_reader_substate = self._state_view.modules.get_absorbance_reader_substate( module_id=params.moduleId ) @@ -62,16 +63,21 @@ async def execute( ) if abs_reader is not None: - result = await abs_reader.start_measure(wavelength=params.sampleWavelength) - converted_values = ( - self._state_view.modules.convert_absorbance_reader_data_points( - data=result + results = await abs_reader.start_measure() + if abs_reader._measurement_config is not None: + asbsorbance_result: Dict[int, Dict[str, float]] = {} + sample_wavelengths = abs_reader._measurement_config.sample_wavelengths + for wavelength, result in zip(sample_wavelengths, results): + converted_values = ( + self._state_view.modules.convert_absorbance_reader_data_points( + data=result + ) + ) + asbsorbance_result[wavelength] = converted_values + return SuccessData( + public=ReadAbsorbanceResult(data=asbsorbance_result), + private=None, ) - ) - return SuccessData( - public=ReadAbsorbanceResult(data=converted_values), - private=None, - ) return SuccessData( public=ReadAbsorbanceResult(data=None), diff --git a/api/src/opentrons/protocol_engine/commands/liquid_probe.py b/api/src/opentrons/protocol_engine/commands/liquid_probe.py index 142cd93aba4..1a8597f9c03 100644 --- a/api/src/opentrons/protocol_engine/commands/liquid_probe.py +++ b/api/src/opentrons/protocol_engine/commands/liquid_probe.py @@ -6,8 +6,12 @@ from pydantic import Field -from opentrons.protocol_engine.errors.exceptions import MustHomeError, TipNotEmptyError from opentrons.protocol_engine.state import update_types +from opentrons.protocol_engine.errors.exceptions import ( + MustHomeError, + PipetteNotReadyToAspirateError, + TipNotEmptyError, +) from opentrons.types import MountType from opentrons_shared_data.errors.exceptions import ( PipetteLiquidNotFoundError, @@ -33,6 +37,7 @@ if TYPE_CHECKING: from ..execution import MovementHandler, PipettingHandler from ..resources import ModelUtils + from ..state.state import StateView LiquidProbeCommandType = Literal["liquidProbe"] @@ -97,7 +102,10 @@ class _ExecuteCommonResult(NamedTuple): async def _execute_common( - movement: MovementHandler, pipetting: PipettingHandler, params: _CommonParams + state_view: StateView, + movement: MovementHandler, + pipetting: PipettingHandler, + params: _CommonParams, ) -> _ExecuteCommonResult: pipette_id = params.pipetteId labware_id = params.labwareId @@ -105,13 +113,21 @@ async def _execute_common( state_update = update_types.StateUpdate() - # _validate_tip_attached in pipetting.py is a private method so we're using - # get_is_ready_to_aspirate as an indirect way to throw a TipNotAttachedError if appropriate - pipetting.get_is_ready_to_aspirate(pipette_id=pipette_id) - - if pipetting.get_is_empty(pipette_id=pipette_id) is False: + # May raise TipNotAttachedError. + aspirated_volume = state_view.pipettes.get_aspirated_volume(pipette_id) + + if aspirated_volume is None: + # Theoretically, we could avoid raising an error by automatically preparing + # to aspirate above the well like AspirateImplementation does. However, the + # only way for this to happen is if someone tries to do a liquid probe with + # a tip that's previously held liquid, which they should avoid anyway. + raise PipetteNotReadyToAspirateError( + "The pipette cannot probe liquid because of a previous blow out." + " The plunger must be reset while the tip is somewhere away from liquid." + ) + elif aspirated_volume != 0: raise TipNotEmptyError( - message="This operation requires a tip with no liquid in it." + message="The pipette cannot probe for liquid when the tip has liquid in it." ) if await movement.check_for_valid_position(mount=MountType.LEFT) is False: @@ -158,11 +174,13 @@ class LiquidProbeImplementation( def __init__( self, + state_view: StateView, movement: MovementHandler, pipetting: PipettingHandler, model_utils: ModelUtils, **kwargs: object, ) -> None: + self._state_view = state_view self._movement = movement self._pipetting = pipetting self._model_utils = model_utils @@ -178,11 +196,13 @@ async def execute(self, params: _CommonParams) -> _LiquidProbeExecuteReturn: the pipette. TipNotEmptyError: as an undefined error, if the tip starts with liquid in it. + PipetteNotReadyToAspirateError: as an undefined error, if the plunger is not + in a safe position to do the liquid probe. MustHomeError: as an undefined error, if the plunger is not in a valid position. """ z_pos_or_error, state_update, deck_point = await _execute_common( - self._movement, self._pipetting, params + self._state_view, self._movement, self._pipetting, params ) if isinstance(z_pos_or_error, PipetteLiquidNotFoundError): return DefinedErrorData( @@ -216,10 +236,12 @@ class TryLiquidProbeImplementation( def __init__( self, + state_view: StateView, movement: MovementHandler, pipetting: PipettingHandler, **kwargs: object, ) -> None: + self._state_view = state_view self._movement = movement self._pipetting = pipetting @@ -231,7 +253,7 @@ async def execute(self, params: _CommonParams) -> _TryLiquidProbeExecuteReturn: of a defined error. """ z_pos_or_error, state_update, deck_point = await _execute_common( - self._movement, self._pipetting, params + self._state_view, self._movement, self._pipetting, params ) z_pos = ( diff --git a/api/src/opentrons/protocol_engine/commands/load_labware.py b/api/src/opentrons/protocol_engine/commands/load_labware.py index 6b040e815da..2de394c482c 100644 --- a/api/src/opentrons/protocol_engine/commands/load_labware.py +++ b/api/src/opentrons/protocol_engine/commands/load_labware.py @@ -17,6 +17,7 @@ from .command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData from ..errors.error_occurrence import ErrorOccurrence +from ..state.update_types import StateUpdate if TYPE_CHECKING: from ..state.state import StateView @@ -141,6 +142,16 @@ async def execute( labware_id=params.labwareId, ) + state_update = StateUpdate() + + state_update.set_loaded_labware( + labware_id=loaded_labware.labware_id, + offset_id=loaded_labware.offsetId, + definition=loaded_labware.definition, + location=verified_location, + display_name=params.displayName, + ) + # TODO(jbl 2023-06-23) these validation checks happen after the labware is loaded, because they rely on # on the definition. In practice this will not cause any issues since they will raise protocol ending # exception, but for correctness should be refactored to do this check beforehand. @@ -157,6 +168,7 @@ async def execute( offsetId=loaded_labware.offsetId, ), private=None, + state_update=state_update, ) diff --git a/api/src/opentrons/protocol_engine/commands/move_labware.py b/api/src/opentrons/protocol_engine/commands/move_labware.py index 463b346457e..965143fdbd7 100644 --- a/api/src/opentrons/protocol_engine/commands/move_labware.py +++ b/api/src/opentrons/protocol_engine/commands/move_labware.py @@ -6,7 +6,6 @@ from typing_extensions import Literal from opentrons.types import Point -from ..state import update_types from ..types import ( CurrentWell, LabwareLocation, @@ -21,6 +20,7 @@ from ..resources import labware_validation, fixture_validation from .command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData from ..errors.error_occurrence import ErrorOccurrence +from ..state.update_types import StateUpdate from opentrons_shared_data.gripper.constants import GRIPPER_PADDLE_WIDTH if TYPE_CHECKING: @@ -98,7 +98,7 @@ async def execute( # noqa: C901 self, params: MoveLabwareParams ) -> SuccessData[MoveLabwareResult, None]: """Move a loaded labware to a new location.""" - state_update = update_types.StateUpdate() + state_update = StateUpdate() # Allow propagation of LabwareNotLoadedError. current_labware = self._state_view.labware.get(labware_id=params.labwareId) @@ -231,6 +231,12 @@ async def execute( # noqa: C901 ): state_update.clear_all_pipette_locations() + state_update.set_labware_location( + labware_id=params.labwareId, + new_location=available_new_location, + new_offset_id=new_offset_id, + ) + return SuccessData( public=MoveLabwareResult(offsetId=new_offset_id), private=None, diff --git a/api/src/opentrons/protocol_engine/commands/reload_labware.py b/api/src/opentrons/protocol_engine/commands/reload_labware.py index 116698552cd..25f545736be 100644 --- a/api/src/opentrons/protocol_engine/commands/reload_labware.py +++ b/api/src/opentrons/protocol_engine/commands/reload_labware.py @@ -6,6 +6,7 @@ from .command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData from ..errors.error_occurrence import ErrorOccurrence +from ..state.update_types import StateUpdate if TYPE_CHECKING: from ..state.state import StateView @@ -64,12 +65,21 @@ async def execute( labware_id=params.labwareId, ) + state_update = StateUpdate() + + state_update.set_labware_location( + labware_id=params.labwareId, + new_location=reloaded_labware.location, + new_offset_id=reloaded_labware.offsetId, + ) + return SuccessData( public=ReloadLabwareResult( labwareId=params.labwareId, offsetId=reloaded_labware.offsetId, ), private=None, + state_update=state_update, ) diff --git a/api/src/opentrons/protocol_engine/execution/pipetting.py b/api/src/opentrons/protocol_engine/execution/pipetting.py index fed6fc52ee6..ae35e779761 100644 --- a/api/src/opentrons/protocol_engine/execution/pipetting.py +++ b/api/src/opentrons/protocol_engine/execution/pipetting.py @@ -30,9 +30,6 @@ class PipettingHandler(TypingProtocol): """Liquid handling commands.""" - def get_is_empty(self, pipette_id: str) -> bool: - """Get whether a pipette has an aspirated volume equal to 0.""" - def get_is_ready_to_aspirate(self, pipette_id: str) -> bool: """Get whether a pipette is ready to aspirate.""" @@ -82,10 +79,6 @@ def __init__(self, state_view: StateView, hardware_api: HardwareControlAPI) -> N self._state_view = state_view self._hardware_api = hardware_api - def get_is_empty(self, pipette_id: str) -> bool: - """Get whether a pipette has an aspirated volume equal to 0.""" - return self._state_view.pipettes.get_aspirated_volume(pipette_id) == 0 - def get_is_ready_to_aspirate(self, pipette_id: str) -> bool: """Get whether a pipette is ready to aspirate.""" hw_pipette = self._state_view.pipettes.get_hardware_pipette( @@ -239,10 +232,6 @@ def __init__( """Initialize a PipettingHandler instance.""" self._state_view = state_view - def get_is_empty(self, pipette_id: str) -> bool: - """Get whether a pipette has an aspirated volume equal to 0.""" - return self._state_view.pipettes.get_aspirated_volume(pipette_id) == 0 - def get_is_ready_to_aspirate(self, pipette_id: str) -> bool: """Get whether a pipette is ready to aspirate.""" return self._state_view.pipettes.get_aspirated_volume(pipette_id) is not None diff --git a/api/src/opentrons/protocol_engine/execution/tip_handler.py b/api/src/opentrons/protocol_engine/execution/tip_handler.py index 3968c7a6923..af6c2fa8c05 100644 --- a/api/src/opentrons/protocol_engine/execution/tip_handler.py +++ b/api/src/opentrons/protocol_engine/execution/tip_handler.py @@ -20,6 +20,8 @@ ProtocolEngineError, ) +from opentrons.hardware_control.nozzle_manager import NozzleConfigurationType + PRIMARY_NOZZLE_TO_ENDING_NOZZLE_MAP = { "A1": {"COLUMN": "H1", "ROW": "A12"}, @@ -300,6 +302,30 @@ async def verify_tip_presence( This function will raise an exception if the specified tip presence status isn't matched. """ + nozzle_configuration = ( + self._state_view.pipettes.state.nozzle_configuration_by_id[pipette_id] + ) + + # Configuration metrics by which tip presence checking is ignored + unsupported_pipette_types = [8, 96] + unsupported_layout_types = [ + NozzleConfigurationType.SINGLE, + NozzleConfigurationType.COLUMN, + ] + # NOTE: (09-20-2024) Current on multi-channel pipettes, utilizing less than 4 nozzles risks false positives on the tip presence sensor + supported_partial_nozzle_minimum = 4 + + if ( + nozzle_configuration is not None + and self._state_view.pipettes.get_channels(pipette_id) + in unsupported_pipette_types + and nozzle_configuration.configuration in unsupported_layout_types + and len(nozzle_configuration.map_store) < supported_partial_nozzle_minimum + ): + # Tip presence sensing is not supported for single tip pick up on the 96ch Flex Pipette, nor with single and some partial layous of the 8ch Flex Pipette. + # This is due in part to a press distance tolerance which creates a risk case for false positives. In the case of single tip, the mechanical tolerance + # for presses with 100% success is below the minimum average achieved press distance for a given multi channel pipette in that configuration. + return try: ot3api = ensure_ot3_hardware(hardware_api=self._hardware_api) hw_mount = self._state_view.pipettes.get_mount(pipette_id).to_hw_mount() diff --git a/api/src/opentrons/protocol_engine/resources/pipette_data_provider.py b/api/src/opentrons/protocol_engine/resources/pipette_data_provider.py index 43b3be16f38..d3998c69bd1 100644 --- a/api/src/opentrons/protocol_engine/resources/pipette_data_provider.py +++ b/api/src/opentrons/protocol_engine/resources/pipette_data_provider.py @@ -193,9 +193,12 @@ def _get_virtual_pipette_static_config_by_model( # noqa: C901 pipette_model.pipette_channels, pipette_model.pipette_version, ) - nozzle_manager = NozzleConfigurationManager.build_from_config( - config, valid_nozzle_maps - ) + if pipette_id not in self._nozzle_manager_layout_by_id: + nozzle_manager = NozzleConfigurationManager.build_from_config( + config, valid_nozzle_maps + ) + else: + nozzle_manager = self._nozzle_manager_layout_by_id[pipette_id] tip_overlap_dict_for_tip_type = None for configuration in ( diff --git a/api/src/opentrons/protocol_engine/state/frustum_helpers.py b/api/src/opentrons/protocol_engine/state/frustum_helpers.py index b78957a2f5f..27e417aa8b4 100644 --- a/api/src/opentrons/protocol_engine/state/frustum_helpers.py +++ b/api/src/opentrons/protocol_engine/state/frustum_helpers.py @@ -1,11 +1,14 @@ """Helper functions for liquid-level related calculations inside a given frustum.""" -from typing import List, Tuple, Iterator, Sequence, Any +from typing import List, Tuple, Iterator, Sequence, Any, Union, Optional from numpy import pi, iscomplex, roots, real +from math import isclose -from ..errors.exceptions import InvalidLiquidHeightFound +from ..errors.exceptions import InvalidLiquidHeightFound, InvalidWellDefinitionError from opentrons_shared_data.labware.types import ( is_circular_frusta_list, is_rectangular_frusta_list, + CircularBoundedSection, + RectangularBoundedSection, ) from opentrons_shared_data.labware.labware_definition import InnerWellGeometry @@ -14,14 +17,15 @@ def reject_unacceptable_heights( potential_heights: List[float], max_height: float ) -> float: """Reject any solutions to a polynomial equation that cannot be the height of a frustum.""" - valid_heights = [] + valid_heights: List[float] = [] for root in potential_heights: # reject any heights that are negative or greater than the max height if not iscomplex(root): # take only the real component of the root and round to 4 decimal places rounded_root = round(real(root), 4) if (rounded_root <= max_height) and (rounded_root >= 0): - valid_heights.append(rounded_root) + if not any([isclose(rounded_root, height) for height in valid_heights]): + valid_heights.append(rounded_root) if len(valid_heights) != 1: raise InvalidLiquidHeightFound( message="Unable to estimate valid liquid height from volume." @@ -29,6 +33,33 @@ def reject_unacceptable_heights( return valid_heights[0] +def get_cross_section_area( + bounded_section: Union[CircularBoundedSection, RectangularBoundedSection] +) -> float: + """Find the shape of a cross-section and calculate the area appropriately.""" + if bounded_section["shape"] == "circular": + cross_section_area = cross_section_area_circular(bounded_section["diameter"]) + elif bounded_section["shape"] == "rectangular": + cross_section_area = cross_section_area_rectangular( + bounded_section["xDimension"], + bounded_section["yDimension"], + ) + else: + raise InvalidWellDefinitionError(message="Invalid well volume components.") + return cross_section_area + + +def cross_section_area_circular(diameter: float) -> float: + """Get the area of a circular cross-section.""" + radius = diameter / 2 + return pi * (radius**2) + + +def cross_section_area_rectangular(x_dimension: float, y_dimension: float) -> float: + """Get the area of a rectangular cross-section.""" + return x_dimension * y_dimension + + def rectangular_frustum_polynomial_roots( bottom_length: float, bottom_width: float, @@ -181,7 +212,7 @@ def height_from_volume_spherical( return height -def get_boundary_cross_sections(frusta: Sequence[Any]) -> Iterator[Tuple[Any, Any]]: +def get_boundary_pairs(frusta: Sequence[Any]) -> Iterator[Tuple[Any, Any]]: """Yield tuples representing two cross-section boundaries of a segment of a well.""" iter_f = iter(frusta) el = next(iter_f) @@ -201,7 +232,7 @@ def get_well_volumetric_capacity( if well_geometry.bottomShape.shape == "spherical": bottom_spherical_section_depth = well_geometry.bottomShape.depth bottom_sphere_volume = volume_from_height_spherical( - radius_of_curvature=well_geometry.bottomShape.radius_of_curvature, + radius_of_curvature=well_geometry.bottomShape.radiusOfCurvature, target_height=bottom_spherical_section_depth, ) well_volume.append((bottom_spherical_section_depth, bottom_sphere_volume)) @@ -210,7 +241,7 @@ def get_well_volumetric_capacity( sorted_frusta = sorted(well_geometry.frusta, key=lambda section: section.topHeight) if is_rectangular_frusta_list(sorted_frusta): - for f, next_f in get_boundary_cross_sections(sorted_frusta): + for f, next_f in get_boundary_pairs(sorted_frusta): top_cross_section_width = next_f["xDimension"] top_cross_section_length = next_f["yDimension"] bottom_cross_section_width = f["xDimension"] @@ -227,7 +258,7 @@ def get_well_volumetric_capacity( well_volume.append((next_f["topHeight"], frustum_volume)) elif is_circular_frusta_list(sorted_frusta): - for f, next_f in get_boundary_cross_sections(sorted_frusta): + for f, next_f in get_boundary_pairs(sorted_frusta): top_cross_section_radius = next_f["diameter"] / 2.0 bottom_cross_section_radius = f["diameter"] / 2.0 frustum_height = next_f["topHeight"] - f["topHeight"] @@ -239,5 +270,201 @@ def get_well_volumetric_capacity( ) well_volume.append((next_f["topHeight"], frustum_volume)) - + else: + raise NotImplementedError( + "Well section with differing boundary shapes not yet implemented." + ) return well_volume + + +def height_at_volume_within_section( + top_cross_section: Union[CircularBoundedSection, RectangularBoundedSection], + bottom_cross_section: Union[CircularBoundedSection, RectangularBoundedSection], + target_volume_relative: float, + frustum_height: float, +) -> float: + """Calculate a height within a bounded section according to geometry.""" + if top_cross_section["shape"] == bottom_cross_section["shape"] == "circular": + frustum_height = height_from_volume_circular( + volume=target_volume_relative, + top_radius=(top_cross_section["diameter"] / 2), + bottom_radius=(bottom_cross_section["diameter"] / 2), + total_frustum_height=frustum_height, + ) + elif top_cross_section["shape"] == bottom_cross_section["shape"] == "rectangular": + frustum_height = height_from_volume_rectangular( + volume=target_volume_relative, + total_frustum_height=frustum_height, + bottom_width=bottom_cross_section["xDimension"], + bottom_length=bottom_cross_section["yDimension"], + top_width=top_cross_section["xDimension"], + top_length=top_cross_section["yDimension"], + ) + else: + raise NotImplementedError( + "Height from volume calculation not yet implemented for this well shape." + ) + return frustum_height + + +def volume_at_height_within_section( + top_cross_section: Union[CircularBoundedSection, RectangularBoundedSection], + bottom_cross_section: Union[CircularBoundedSection, RectangularBoundedSection], + target_height_relative: float, + frustum_height: float, +) -> float: + """Calculate a volume within a bounded section according to geometry.""" + if top_cross_section["shape"] == bottom_cross_section["shape"] == "circular": + frustum_volume = volume_from_height_circular( + target_height=target_height_relative, + total_frustum_height=frustum_height, + bottom_radius=(bottom_cross_section["diameter"] / 2), + top_radius=(top_cross_section["diameter"] / 2), + ) + elif top_cross_section["shape"] == bottom_cross_section["shape"] == "rectangular": + frustum_volume = volume_from_height_rectangular( + target_height=target_height_relative, + total_frustum_height=frustum_height, + bottom_width=bottom_cross_section["xDimension"], + bottom_length=bottom_cross_section["yDimension"], + top_width=top_cross_section["xDimension"], + top_length=top_cross_section["yDimension"], + ) + # TODO(cm): this would be the NEST-96 2uL wells referenced in EXEC-712 + # we need to input the math attached to that issue + else: + raise NotImplementedError( + "Height from volume calculation not yet implemented for this well shape." + ) + return frustum_volume + + +def _find_volume_in_partial_frustum( + sorted_frusta: List[Any], + target_height: float, +) -> Optional[float]: + """Look through a sorted list of frusta for a target height, and find the volume at that height.""" + partial_volume: Optional[float] = None + for bottom_cross_section, top_cross_section in get_boundary_pairs(sorted_frusta): + if ( + bottom_cross_section["topHeight"] + < target_height + < top_cross_section["targetHeight"] + ): + relative_target_height = target_height - bottom_cross_section["topHeight"] + frustum_height = ( + top_cross_section["topHeight"] - bottom_cross_section["topHeight"] + ) + partial_volume = volume_at_height_within_section( + top_cross_section=top_cross_section, + bottom_cross_section=bottom_cross_section, + target_height_relative=relative_target_height, + frustum_height=frustum_height, + ) + return partial_volume + + +def find_volume_at_well_height( + target_height: float, well_geometry: InnerWellGeometry +) -> float: + """Find the volume within a well, at a known height.""" + volumetric_capacity = get_well_volumetric_capacity(well_geometry) + max_height = volumetric_capacity[-1][0] + if target_height < 0 or target_height > max_height: + raise InvalidLiquidHeightFound("Invalid target height.") + # volumes in volumetric_capacity are relative to each frustum, + # so we have to find the volume of all the full sections enclosed + # beneath the target height + closed_section_volume = 0.0 + for boundary_height, section_volume in volumetric_capacity: + if boundary_height > target_height: + break + closed_section_volume += section_volume + # if target height is a boundary cross-section, we already know the volume + if target_height == boundary_height: + return closed_section_volume + # find the section the target height is in and compute the volume + # since bottomShape is not in list of frusta, check here first + if well_geometry.bottomShape: + bottom_segment_height = volumetric_capacity[0][0] + if ( + target_height < bottom_segment_height + and well_geometry.bottomShape.shape == "spherical" + ): + return volume_from_height_spherical( + target_height=target_height, + radius_of_curvature=well_geometry.bottomShape.radiusOfCurvature, + ) + sorted_frusta = sorted(well_geometry.frusta, key=lambda section: section.topHeight) + # TODO(cm): handle non-frustum section that is not at the bottom. + partial_volume = _find_volume_in_partial_frustum( + sorted_frusta=sorted_frusta, + target_height=target_height, + ) + if not partial_volume: + raise InvalidLiquidHeightFound("Unable to find volume at given well-height.") + return partial_volume + closed_section_volume + + +def _find_height_in_partial_frustum( + sorted_frusta: List[Any], + volumetric_capacity: List[Tuple[float, float]], + target_volume: float, +) -> Optional[float]: + """Look through a sorted list of frusta for a target volume, and find the height at that volume.""" + well_height: Optional[float] = None + for cross_sections, capacity in zip( + get_boundary_pairs(sorted_frusta), + get_boundary_pairs(volumetric_capacity), + ): + bottom_cross_section, top_cross_section = cross_sections + (bottom_height, bottom_volume), (top_height, top_volume) = capacity + + if bottom_volume < target_volume < top_volume: + relative_target_volume = target_volume - bottom_volume + frustum_height = top_height - bottom_height + partial_height = height_at_volume_within_section( + top_cross_section=top_cross_section, + bottom_cross_section=bottom_cross_section, + target_volume_relative=relative_target_volume, + frustum_height=frustum_height, + ) + well_height = partial_height + bottom_height + return well_height + + +def find_height_at_well_volume( + target_volume: float, well_geometry: InnerWellGeometry +) -> float: + """Find the height within a well, at a known volume.""" + volumetric_capacity = get_well_volumetric_capacity(well_geometry) + max_volume = volumetric_capacity[-1][1] + if target_volume < 0 or target_volume > max_volume: + raise InvalidLiquidHeightFound("Invalid target volume.") + + sorted_frusta = sorted(well_geometry.frusta, key=lambda section: section.topHeight) + # find the section the target volume is in and compute the height + # since bottomShape is not in list of frusta, check here first + if well_geometry.bottomShape: + volume_within_bottom_segment = volumetric_capacity[0][1] + if ( + target_volume < volume_within_bottom_segment + and well_geometry.bottomShape.shape == "spherical" + ): + return height_from_volume_spherical( + volume=target_volume, + radius_of_curvature=well_geometry.bottomShape.radiusOfCurvature, + total_frustum_height=well_geometry.bottomShape.depth, + ) + # if bottom shape is present but doesn't contain the target volume, + # then we need to look through the volumetric capacity list without the bottom shape + # so volumetric_capacity and sorted_frusta will be aligned + volumetric_capacity.pop(0) + well_height = _find_height_in_partial_frustum( + sorted_frusta=sorted_frusta, + volumetric_capacity=volumetric_capacity, + target_volume=target_volume, + ) + if not well_height: + raise InvalidLiquidHeightFound("Unable to find height at given well-volume.") + return well_height diff --git a/api/src/opentrons/protocol_engine/state/geometry.py b/api/src/opentrons/protocol_engine/state/geometry.py index a0fef65e7ee..502f0d4d8eb 100644 --- a/api/src/opentrons/protocol_engine/state/geometry.py +++ b/api/src/opentrons/protocol_engine/state/geometry.py @@ -53,7 +53,11 @@ from .modules import ModuleView from .pipettes import PipetteView from .addressable_areas import AddressableAreaView -from .frustum_helpers import get_well_volumetric_capacity +from .frustum_helpers import ( + get_well_volumetric_capacity, + find_volume_at_well_height, + find_height_at_well_volume, +) SLOT_WIDTH = 128 @@ -1218,3 +1222,35 @@ def get_well_volumetric_capacity( message=f"No InnerWellGeometry found for well id: {well_id}" ) return get_well_volumetric_capacity(well_geometry) + + def get_volume_at_height( + self, labware_id: str, well_id: str, target_height: float + ) -> float: + """Find the volume at any height within a well.""" + labware_def = self._labware.get_definition(labware_id) + if labware_def.innerLabwareGeometry is None: + raise InvalidWellDefinitionError(message="No InnerLabwareGeometry found.") + well_geometry = labware_def.innerLabwareGeometry.get(well_id) + if well_geometry is None: + raise InvalidWellDefinitionError( + message=f"No InnerWellGeometry found for well id: {well_id}" + ) + return find_volume_at_well_height( + target_height=target_height, well_geometry=well_geometry + ) + + def get_height_at_volume( + self, labware_id: str, well_id: str, target_volume: float + ) -> float: + """Find the height from any volume in a well.""" + labware_def = self._labware.get_definition(labware_id) + if labware_def.innerLabwareGeometry is None: + raise InvalidWellDefinitionError(message="No InnerLabwareGeometry found.") + well_geometry = labware_def.innerLabwareGeometry.get(well_id) + if well_geometry is None: + raise InvalidWellDefinitionError( + message=f"No InnerWellGeometry found for well id: {well_id}" + ) + return find_height_at_well_volume( + target_volume=target_volume, well_geometry=well_geometry + ) diff --git a/api/src/opentrons/protocol_engine/state/labware.py b/api/src/opentrons/protocol_engine/state/labware.py index c7f11abb7ec..0db6b310e1e 100644 --- a/api/src/opentrons/protocol_engine/state/labware.py +++ b/api/src/opentrons/protocol_engine/state/labware.py @@ -15,6 +15,7 @@ Union, ) +from opentrons.protocol_engine.state import update_types from opentrons_shared_data.deck.types import DeckDefinitionV5 from opentrons_shared_data.gripper.constants import LABWARE_GRIP_FORCE from opentrons_shared_data.labware.labware_definition import LabwareRole @@ -27,13 +28,6 @@ from .. import errors from ..resources import DeckFixedLabware, labware_validation, fixture_validation -from ..commands import ( - Command, - absorbance_reader, - LoadLabwareResult, - MoveLabwareResult, - ReloadLabwareResult, -) from ..types import ( DeckSlotLocation, OnLabwareLocation, @@ -155,7 +149,7 @@ def __init__( def handle_action(self, action: Action) -> None: """Modify state in reaction to an action.""" if isinstance(action, SucceedCommandAction): - self._handle_command(action.command) + self._handle_command(action) elif isinstance(action, AddLabwareOffsetAction): labware_offset = LabwareOffset.construct( @@ -175,74 +169,82 @@ def handle_action(self, action: Action) -> None: ) self._state.definitions_by_uri[uri] = action.definition - def _handle_command(self, command: Command) -> None: + def _handle_command(self, action: Action) -> None: """Modify state in reaction to a command.""" - if isinstance(command.result, LoadLabwareResult): + self._add_loaded_labware(action) + self._set_labware_location(action) + + def _add_labware_offset(self, labware_offset: LabwareOffset) -> None: + """Add a new labware offset to state. + + `labware_offset.id` must not match any existing labware offset ID. + `LoadLabwareCommand`s retain references to their corresponding labware offsets + and expect them to be immutable. + """ + assert labware_offset.id not in self._state.labware_offsets_by_id + + self._state.labware_offsets_by_id[labware_offset.id] = labware_offset + + def _add_loaded_labware(self, action: Action) -> None: + if ( + isinstance(action, SucceedCommandAction) + and action.state_update.loaded_labware != update_types.NO_CHANGE + ): # If the labware load refers to an offset, that offset must actually exist. - if command.result.offsetId is not None: - assert command.result.offsetId in self._state.labware_offsets_by_id + if action.state_update.loaded_labware.offset_id is not None: + assert ( + action.state_update.loaded_labware.offset_id + in self._state.labware_offsets_by_id + ) definition_uri = uri_from_details( - namespace=command.result.definition.namespace, - load_name=command.result.definition.parameters.loadName, - version=command.result.definition.version, + namespace=action.state_update.loaded_labware.definition.namespace, + load_name=action.state_update.loaded_labware.definition.parameters.loadName, + version=action.state_update.loaded_labware.definition.version, ) - self._state.definitions_by_uri[definition_uri] = command.result.definition - if isinstance(command.result, LoadLabwareResult): - location = command.params.location - else: - location = self._state.labware_by_id[command.result.labwareId].location + self._state.definitions_by_uri[ + definition_uri + ] = action.state_update.loaded_labware.definition + + location = action.state_update.loaded_labware.new_location + + display_name = action.state_update.loaded_labware.display_name self._state.labware_by_id[ - command.result.labwareId + action.state_update.loaded_labware.labware_id ] = LoadedLabware.construct( - id=command.result.labwareId, + id=action.state_update.loaded_labware.labware_id, location=location, - loadName=command.result.definition.parameters.loadName, + loadName=action.state_update.loaded_labware.definition.parameters.loadName, definitionUri=definition_uri, - offsetId=command.result.offsetId, - displayName=command.params.displayName, + offsetId=action.state_update.loaded_labware.offset_id, + displayName=display_name, ) - elif isinstance(command.result, ReloadLabwareResult): - labware_id = command.params.labwareId - new_offset_id = command.result.offsetId - self._state.labware_by_id[labware_id].offsetId = new_offset_id + def _set_labware_location(self, action: Action) -> None: + if ( + isinstance(action, SucceedCommandAction) + and action.state_update.labware_location != update_types.NO_CHANGE + ): - elif isinstance(command.result, MoveLabwareResult): - labware_id = command.params.labwareId - new_location = command.params.newLocation - new_offset_id = command.result.offsetId + labware_id = action.state_update.labware_location.labware_id + new_offset_id = action.state_update.labware_location.offset_id self._state.labware_by_id[labware_id].offsetId = new_offset_id - if isinstance( - new_location, AddressableAreaLocation - ) and fixture_validation.is_gripper_waste_chute( - new_location.addressableAreaName - ): - # If a labware has been moved into a waste chute it's been chuted away and is now technically off deck - new_location = OFF_DECK_LOCATION - self._state.labware_by_id[labware_id].location = new_location - elif isinstance(command.result, absorbance_reader.MoveLidResult): - lid_id = command.result.lidId - new_location = command.result.newLocation - new_offset_id = command.result.offsetId + if action.state_update.labware_location.new_location: + new_location = action.state_update.labware_location.new_location - self._state.labware_by_id[lid_id].offsetId = new_offset_id - self._state.labware_by_id[lid_id].location = new_location + if isinstance( + new_location, AddressableAreaLocation + ) and fixture_validation.is_gripper_waste_chute( + new_location.addressableAreaName + ): + # If a labware has been moved into a waste chute it's been chuted away and is now technically off deck + new_location = OFF_DECK_LOCATION - def _add_labware_offset(self, labware_offset: LabwareOffset) -> None: - """Add a new labware offset to state. - - `labware_offset.id` must not match any existing labware offset ID. - `LoadLabwareCommand`s retain references to their corresponding labware offsets - and expect them to be immutable. - """ - assert labware_offset.id not in self._state.labware_offsets_by_id - - self._state.labware_offsets_by_id[labware_offset.id] = labware_offset + self._state.labware_by_id[labware_id].location = new_location class LabwareView(HasState[LabwareState]): diff --git a/api/src/opentrons/protocol_engine/state/module_substates/absorbance_reader_substate.py b/api/src/opentrons/protocol_engine/state/module_substates/absorbance_reader_substate.py index 1b92948fc90..33b96aa0881 100644 --- a/api/src/opentrons/protocol_engine/state/module_substates/absorbance_reader_substate.py +++ b/api/src/opentrons/protocol_engine/state/module_substates/absorbance_reader_substate.py @@ -1,11 +1,12 @@ """Heater-Shaker Module sub-state.""" from dataclasses import dataclass -from typing import NewType, Optional, Dict +from typing import List, NewType, Optional, Dict from opentrons.protocol_engine.errors import CannotPerformModuleAction AbsorbanceReaderId = NewType("AbsorbanceReaderId", str) AbsorbanceReaderLidId = NewType("AbsorbanceReaderLidId", str) +AbsorbanceReaderMeasureMode = NewType("AbsorbanceReaderMeasureMode", str) @dataclass(frozen=True) @@ -16,8 +17,10 @@ class AbsorbanceReaderSubState: configured: bool measured: bool is_lid_on: bool - data: Optional[Dict[str, float]] - configured_wavelength: Optional[int] + data: Optional[Dict[int, Dict[str, float]]] + configured_wavelengths: Optional[List[int]] + measure_mode: Optional[AbsorbanceReaderMeasureMode] + reference_wavelength: Optional[int] lid_id: Optional[str] def raise_if_lid_status_not_expected(self, lid_on_expected: bool) -> None: diff --git a/api/src/opentrons/protocol_engine/state/modules.py b/api/src/opentrons/protocol_engine/state/modules.py index 3327020f93e..7120025da89 100644 --- a/api/src/opentrons/protocol_engine/state/modules.py +++ b/api/src/opentrons/protocol_engine/state/modules.py @@ -29,6 +29,9 @@ from opentrons.protocol_engine.commands.calibration.calibrate_module import ( CalibrateModuleResult, ) +from opentrons.protocol_engine.state.module_substates.absorbance_reader_substate import ( + AbsorbanceReaderMeasureMode, +) from opentrons.types import DeckSlotName, MountType from ..errors import ModuleNotConnectedError @@ -319,7 +322,9 @@ def _update_absorbance_reader_lid_id( measured=prev_state.measured, is_lid_on=prev_state.is_lid_on, data=prev_state.data, - configured_wavelength=prev_state.configured_wavelength, + measure_mode=prev_state.measure_mode, + configured_wavelengths=prev_state.configured_wavelengths, + reference_wavelength=prev_state.reference_wavelength, lid_id=lid_id, ) @@ -382,29 +387,27 @@ def _add_module_substate( # noqa: C901 module_id=MagneticBlockId(module_id) ) elif ModuleModel.is_absorbance_reader(actual_model): + lid_labware_id = None slot = self._state.slot_by_module_id[module_id] if slot is not None: reader_addressable_area = f"absorbanceReaderV1{slot.value}" - lid_labware_id = None for labware in self._state.deck_fixed_labware: if labware.location == AddressableAreaLocation( addressableAreaName=reader_addressable_area ): lid_labware_id = labware.labware_id break - self._state.substate_by_module_id[module_id] = AbsorbanceReaderSubState( - module_id=AbsorbanceReaderId(module_id), - configured=False, - measured=False, - is_lid_on=True, - data=None, - configured_wavelength=None, - lid_id=lid_labware_id, - ) - else: - raise errors.ModuleNotOnDeckError( - "Opentrons Plate Reader location did not return a valid Deck Slot." - ) + self._state.substate_by_module_id[module_id] = AbsorbanceReaderSubState( + module_id=AbsorbanceReaderId(module_id), + configured=False, + measured=False, + is_lid_on=True, + data=None, + measure_mode=None, + configured_wavelengths=None, + reference_wavelength=None, + lid_id=lid_labware_id, + ) def _update_additional_slots_occupied_by_thermocycler( self, @@ -610,7 +613,9 @@ def _handle_absorbance_reader_commands( # Get current values configured = absorbance_reader_substate.configured - configured_wavelength = absorbance_reader_substate.configured_wavelength + measure_mode = absorbance_reader_substate.measure_mode + configured_wavelengths = absorbance_reader_substate.configured_wavelengths + reference_wavelength = absorbance_reader_substate.reference_wavelength is_lid_on = absorbance_reader_substate.is_lid_on lid_id = absorbance_reader_substate.lid_id data = absorbance_reader_substate.data @@ -621,41 +626,49 @@ def _handle_absorbance_reader_commands( configured=True, measured=False, is_lid_on=is_lid_on, - data=None, - configured_wavelength=command.params.sampleWavelength, lid_id=lid_id, + measure_mode=AbsorbanceReaderMeasureMode(command.params.measureMode), + configured_wavelengths=command.params.sampleWavelengths, + reference_wavelength=command.params.referenceWavelength, + data=None, ) elif isinstance(command.result, absorbance_reader.ReadAbsorbanceResult): self._state.substate_by_module_id[module_id] = AbsorbanceReaderSubState( module_id=AbsorbanceReaderId(module_id), configured=configured, - configured_wavelength=configured_wavelength, - is_lid_on=is_lid_on, measured=True, - data=command.result.data, + is_lid_on=is_lid_on, lid_id=lid_id, + measure_mode=measure_mode, + configured_wavelengths=configured_wavelengths, + reference_wavelength=reference_wavelength, + data=command.result.data, ) elif isinstance(command.result, absorbance_reader.OpenLidResult): self._state.substate_by_module_id[module_id] = AbsorbanceReaderSubState( module_id=AbsorbanceReaderId(module_id), configured=configured, - configured_wavelength=configured_wavelength, - is_lid_on=False, measured=True, - data=data, + is_lid_on=False, lid_id=lid_id, + measure_mode=measure_mode, + configured_wavelengths=configured_wavelengths, + reference_wavelength=reference_wavelength, + data=data, ) elif isinstance(command.result, absorbance_reader.CloseLidResult): self._state.substate_by_module_id[module_id] = AbsorbanceReaderSubState( module_id=AbsorbanceReaderId(module_id), configured=configured, - configured_wavelength=configured_wavelength, - is_lid_on=True, measured=True, - data=data, + is_lid_on=True, lid_id=lid_id, + measure_mode=measure_mode, + configured_wavelengths=configured_wavelengths, + reference_wavelength=reference_wavelength, + data=data, ) diff --git a/api/src/opentrons/protocol_engine/state/pipettes.py b/api/src/opentrons/protocol_engine/state/pipettes.py index 7c77bd979e4..6f942281906 100644 --- a/api/src/opentrons/protocol_engine/state/pipettes.py +++ b/api/src/opentrons/protocol_engine/state/pipettes.py @@ -16,6 +16,7 @@ from opentrons_shared_data.pipette import pipette_definition from opentrons.config.defaults_ot2 import Z_RETRACT_DISTANCE from opentrons.hardware_control.dev_types import PipetteDict +from opentrons.hardware_control import CriticalPoint from opentrons.hardware_control.nozzle_manager import ( NozzleConfigurationType, NozzleMap, @@ -474,7 +475,7 @@ def get_aspirated_volume(self, pipette_id: str) -> Optional[float]: Returns: The volume the pipette has aspirated. - None, after blow-out and the plunger is in an unsafe position or drop-tip and there is no tip attached. + None, after blow-out and the plunger is in an unsafe position. Raises: PipetteNotLoadedError: pipette ID does not exist. @@ -659,17 +660,27 @@ def get_primary_nozzle(self, pipette_id: str) -> Optional[str]: nozzle_map = self._state.nozzle_configuration_by_id.get(pipette_id) return nozzle_map.starting_nozzle if nozzle_map else None - def get_primary_nozzle_offset(self, pipette_id: str) -> Point: - """Get the pipette's current primary nozzle's offset.""" + def _get_critical_point_offset_without_tip( + self, pipette_id: str, critical_point: Optional[CriticalPoint] + ) -> Point: + """Get the offset of the specified critical point from pipette's mount position.""" nozzle_map = self._state.nozzle_configuration_by_id.get(pipette_id) - if nozzle_map: - primary_nozzle_offset = nozzle_map.starting_nozzle_offset - else: - # When not in partial configuration, back-left nozzle is the primary - primary_nozzle_offset = self.get_config( - pipette_id - ).bounding_nozzle_offsets.back_left_offset - return primary_nozzle_offset + # Nozzle map is unavailable only when there's no pipette loaded + # so this is merely for satisfying the type checker + assert ( + nozzle_map is not None + ), "Error getting critical point offset. Nozzle map not found." + match critical_point: + case CriticalPoint.INSTRUMENT_XY_CENTER: + return nozzle_map.instrument_xy_center_offset + case CriticalPoint.XY_CENTER: + return nozzle_map.xy_center_offset + case CriticalPoint.Y_CENTER: + return nozzle_map.y_center_offset + case CriticalPoint.FRONT_NOZZLE: + return nozzle_map.front_nozzle_offset + case _: + return nozzle_map.starting_nozzle_offset def get_pipette_bounding_nozzle_offsets( self, pipette_id: str @@ -681,32 +692,46 @@ def get_pipette_bounding_box(self, pipette_id: str) -> PipetteBoundingBoxOffsets """Get the bounding box of the pipette.""" return self.get_config(pipette_id).pipette_bounding_box_offsets + # TODO (spp, 2024-09-17): in order to find the position of pipette at destination, + # this method repeats the same steps that waypoints builder does while finding + # waypoints to move to. We should consolidate these steps into a shared entity + # so that the deck conflict checker and movement plan builder always remain in sync. def get_pipette_bounds_at_specified_move_to_position( self, pipette_id: str, destination_position: Point, + critical_point: Optional[CriticalPoint], ) -> Tuple[Point, Point, Point, Point]: - """Get the pipette's bounding offsets when primary nozzle is at the given position.""" - primary_nozzle_offset = self.get_primary_nozzle_offset(pipette_id) + """Get the pipette's bounding box position when critical point is at the destination position. + + Returns a tuple of the pipette's bounding box position in deck coordinates as- + (back_left_bound, front_right_bound, back_right_bound, front_left_bound) + Bounding box of the pipette includes the pipette's outer casing as well as nozzles. + """ tip = self.get_attached_tip(pipette_id) - # TODO update this for pipette robot stackup - # Primary nozzle position at destination, in deck coordinates - primary_nozzle_position = destination_position + Point( + + # *Offset* of pipette's critical point w.r.t pipette mount + critical_point_offset = self._get_critical_point_offset_without_tip( + pipette_id, critical_point + ) + + # Position of the above critical point at destination, in deck coordinates + critical_point_position = destination_position + Point( x=0, y=0, z=tip.length if tip else 0 ) - # Get the pipette bounding box based on total nozzles + # Get the pipette bounding box coordinates pipette_bounds_offsets = self.get_config( pipette_id ).pipette_bounding_box_offsets pip_back_left_bound = ( - primary_nozzle_position - - primary_nozzle_offset + critical_point_position + - critical_point_offset + pipette_bounds_offsets.back_left_corner ) pip_front_right_bound = ( - primary_nozzle_position - - primary_nozzle_offset + critical_point_position + - critical_point_offset + pipette_bounds_offsets.front_right_corner ) pip_back_right_bound = Point( diff --git a/api/src/opentrons/protocol_engine/state/update_types.py b/api/src/opentrons/protocol_engine/state/update_types.py index 230e967bb07..6d5e28e3e6b 100644 --- a/api/src/opentrons/protocol_engine/state/update_types.py +++ b/api/src/opentrons/protocol_engine/state/update_types.py @@ -5,7 +5,8 @@ import enum import typing -from opentrons.protocol_engine.types import DeckPoint +from opentrons.protocol_engine.types import DeckPoint, LabwareLocation +from opentrons_shared_data.labware.labware_definition import LabwareDefinition class _NoChangeEnum(enum.Enum): @@ -75,12 +76,37 @@ class PipetteLocationUpdate: new_deck_point: DeckPoint | NoChangeType +@dataclasses.dataclass +class LabwareLocationUpdate: + """Represents an update to perform on a labware's location.""" + + labware_id: str + + new_location: LabwareLocation + """The labware's new logical location.""" + + offset_id: typing.Optional[str] + + +@dataclasses.dataclass +class LoadedLabwareUpdate(LabwareLocationUpdate): + """Update loaded labware.""" + + display_name: typing.Optional[str] + + definition: LabwareDefinition + + @dataclasses.dataclass class StateUpdate: """Represents an update to perform on engine state.""" pipette_location: PipetteLocationUpdate | NoChangeType | ClearType = NO_CHANGE + labware_location: LabwareLocationUpdate | NoChangeType = NO_CHANGE + + loaded_labware: LoadedLabwareUpdate | NoChangeType = NO_CHANGE + # These convenience functions let the caller avoid the boilerplate of constructing a # complicated dataclass tree, and they give us a @@ -134,6 +160,36 @@ def set_pipette_location( # noqa: D102 new_deck_point=new_deck_point, ) + def set_labware_location( # noqa: D102 + self, + *, + labware_id: str, + new_location: LabwareLocation, + new_offset_id: str | None, + ) -> None: + self.labware_location = LabwareLocationUpdate( + labware_id=labware_id, + new_location=new_location, + offset_id=new_offset_id, + ) + + def set_loaded_labware( + self, + definition: LabwareDefinition, + labware_id: str, + offset_id: typing.Optional[str], + display_name: typing.Optional[str], + location: LabwareLocation, + ) -> None: + """Add loaded labware to state.""" + self.loaded_labware = LoadedLabwareUpdate( + definition=definition, + labware_id=labware_id, + offset_id=offset_id, + new_location=location, + display_name=display_name, + ) + def clear_all_pipette_locations(self) -> None: """Mark all pipettes as having an unknown location.""" self.pipette_location = CLEAR diff --git a/api/src/opentrons/protocol_engine/types.py b/api/src/opentrons/protocol_engine/types.py index 519d39b6ec7..e011b43fd7f 100644 --- a/api/src/opentrons/protocol_engine/types.py +++ b/api/src/opentrons/protocol_engine/types.py @@ -1089,3 +1089,6 @@ class CSVParameter(RTPBase): CSVRunTimeParamFilesType = Mapping[StrictStr, StrictStr] CSVRuntimeParamPaths = Dict[str, Path] + + +ABSMeasureMode = Literal["single", "multi"] diff --git a/api/src/opentrons/protocol_runner/legacy_command_mapper.py b/api/src/opentrons/protocol_runner/legacy_command_mapper.py index 2e46e64663c..8ffb67d3560 100644 --- a/api/src/opentrons/protocol_runner/legacy_command_mapper.py +++ b/api/src/opentrons/protocol_runner/legacy_command_mapper.py @@ -34,7 +34,9 @@ ModuleDataProvider, pipette_data_provider, ) -from opentrons.protocol_engine.state.update_types import StateUpdate +from opentrons.protocol_engine.state.update_types import ( + StateUpdate, +) from opentrons_shared_data.labware.labware_definition import LabwareDefinition from opentrons_shared_data.errors import ErrorCodes, EnumeratedError, PythonException @@ -675,10 +677,20 @@ def _map_labware_load( # We just set this above, so we know it's not None. started_at=succeeded_command.startedAt, # type: ignore[arg-type] ) + state_update = StateUpdate() + assert succeeded_command.result is not None + state_update.set_loaded_labware( + labware_id=labware_id, + definition=succeeded_command.result.definition, + display_name=labware_load_info.labware_display_name, + offset_id=labware_load_info.offset_id, + location=location, + ) + succeed_action = pe_actions.SucceedCommandAction( command=succeeded_command, private_result=None, - state_update=StateUpdate(), + state_update=state_update, ) self._command_count["LOAD_LABWARE"] = count + 1 diff --git a/api/tests/opentrons/drivers/absorbance_reader/test_driver.py b/api/tests/opentrons/drivers/absorbance_reader/test_driver.py index ff633e38760..0284f277e2c 100644 --- a/api/tests/opentrons/drivers/absorbance_reader/test_driver.py +++ b/api/tests/opentrons/drivers/absorbance_reader/test_driver.py @@ -1,3 +1,4 @@ +from typing import cast from mock import MagicMock import pytest import asyncio @@ -9,7 +10,7 @@ AbsorbanceHidInterface, ) from opentrons.drivers.absorbance_reader.async_byonoy import AsyncByonoy -from opentrons.drivers.types import AbsorbanceReaderLidStatus +from opentrons.drivers.types import ABSMeasurementMode, AbsorbanceReaderLidStatus @pytest.fixture @@ -23,8 +24,8 @@ def mock_device() -> MagicMock: class MockErrorCode(Enum): - BYONOY_ERROR_NO_ERROR = "no_error" - BYONOY_ERROR = "error" + NO_ERROR = "no_error" + ERROR = "error" @pytest.fixture @@ -52,20 +53,20 @@ async def test_driver_connect_disconnect( mock_interface: MagicMock, driver: AbsorbanceReaderDriver, ) -> None: - mock_interface.byonoy_open_device.return_value = ( - MockErrorCode.BYONOY_ERROR_NO_ERROR, + mock_interface.open_device.return_value = ( + MockErrorCode.NO_ERROR, 1, ) assert not await driver.is_connected() await driver.connect() - mock_interface.byonoy_open_device.assert_called_once() + mock_interface.open_device.assert_called_once() assert await driver.is_connected() assert driver._connection._verify_device_handle() assert driver._connection._device_handle == 1 - mock_interface.byonoy_free_device.return_value = MockErrorCode.BYONOY_ERROR_NO_ERROR + mock_interface.free_device.return_value = MockErrorCode.NO_ERROR await driver.disconnect() assert not await driver.is_connected() @@ -78,18 +79,18 @@ async def test_driver_get_device_info( ) -> None: DEVICE_INFO = MagicMock(AbsorbanceHidInterface.DeviceInfo) - DEVICE_INFO.ref_no = "" - DEVICE_INFO.sn = "SN: BYOMAA00013 REF: DE MAA 001" + DEVICE_INFO.ref_no = "DE MAA 001" + DEVICE_INFO.sn = "BYOMAA00013" DEVICE_INFO.version = "Absorbance V1.0.2 2024-04-18" - mock_interface.byonoy_get_device_information.return_value = ( - MockErrorCode.BYONOY_ERROR_NO_ERROR, + mock_interface.get_device_information.return_value = ( + MockErrorCode.NO_ERROR, DEVICE_INFO, ) info = await connected_driver.get_device_info() - mock_interface.byonoy_get_device_information.assert_called_once() + mock_interface.get_device_information.assert_called_once() assert info == {"serial": "BYOMAA00013", "model": "ABS96", "version": "v1.0.2"} @@ -104,14 +105,14 @@ async def test_driver_get_lid_status( module_status: AbsorbanceReaderLidStatus, ) -> None: - mock_interface.byonoy_get_device_parts_aligned.return_value = ( - MockErrorCode.BYONOY_ERROR_NO_ERROR, + mock_interface.get_device_parts_aligned.return_value = ( + MockErrorCode.NO_ERROR, parts_aligned, ) status = await connected_driver.get_lid_status() - mock_interface.byonoy_get_device_parts_aligned.assert_called_once() + mock_interface.get_device_parts_aligned.assert_called_once() assert status == module_status @@ -120,8 +121,8 @@ async def test_driver_get_supported_wavelengths( connected_driver: AbsorbanceReaderDriver, ) -> None: SUPPORTED_WAVELENGTHS = [450, 500] - mock_interface.byonoy_abs96_get_available_wavelengths.return_value = ( - MockErrorCode.BYONOY_ERROR_NO_ERROR, + mock_interface.abs96_get_available_wavelengths.return_value = ( + MockErrorCode.NO_ERROR, SUPPORTED_WAVELENGTHS, ) @@ -129,42 +130,91 @@ async def test_driver_get_supported_wavelengths( wavelengths = await connected_driver.get_available_wavelengths() - mock_interface.byonoy_abs96_get_available_wavelengths.assert_called_once() + mock_interface.abs96_get_available_wavelengths.assert_called_once() assert connected_driver._connection._supported_wavelengths == SUPPORTED_WAVELENGTHS assert wavelengths == SUPPORTED_WAVELENGTHS -async def test_driver_initialize_and_read( +async def test_driver_initialize_and_read_single( mock_interface: MagicMock, connected_driver: AbsorbanceReaderDriver, ) -> None: # set up mock interface connected_driver._connection._supported_wavelengths = [450, 500] - mock_interface.byonoy_abs96_initialize_single_measurement.return_value = ( - MockErrorCode.BYONOY_ERROR_NO_ERROR - ) - mock_interface.ByonoyAbs96SingleMeasurementConfig = MagicMock( - spec=AbsorbanceHidInterface.MeasurementConfig + mock_interface.abs96_initialize_single_measurement.return_value = ( + MockErrorCode.NO_ERROR ) + class MeasurementConfig(AbsorbanceHidInterface.SingleMeasurementConfig): + def __init__(self) -> None: + self.sample_wavelength = 0 + self.reference_wavelength = 0 + + mock_interface.Abs96SingleMeasurementConfig = MeasurementConfig + # current config should not have been setup yet assert not connected_driver._connection._current_config - await connected_driver.initialize_measurement(450) + await connected_driver.initialize_measurement([450], mode=ABSMeasurementMode.SINGLE) - conf = connected_driver._connection._current_config + conf = cast( + AbsorbanceHidInterface.SingleMeasurementConfig, + connected_driver._connection._current_config, + ) assert conf and conf.sample_wavelength == 450 - mock_interface.byonoy_abs96_initialize_single_measurement.assert_called_once_with( + mock_interface.abs96_initialize_single_measurement.assert_called_once_with(1, conf) + + # setup up mock interface with a single reading + MEASURE_RESULT = [[0.1] * 96] + mock_interface.abs96_single_measure.return_value = ( + MockErrorCode.NO_ERROR, + MEASURE_RESULT, + ) + + result = await connected_driver.get_measurement() + mock_interface.abs96_single_measure.assert_called_once_with(1, conf) + + assert result == MEASURE_RESULT + + +async def test_driver_initialize_and_read_multi( + mock_interface: MagicMock, + connected_driver: AbsorbanceReaderDriver, +) -> None: + # set up mock interface + connected_driver._connection._supported_wavelengths = [450, 500, 600] + mock_interface.abs96_initialize_multiple_measurement.return_value = ( + MockErrorCode.NO_ERROR + ) + + class MeasurementConfig(AbsorbanceHidInterface.MultiMeasurementConfig): + def __init__(self) -> None: + self.sample_wavelengths = [0] + + mock_interface.Abs96MultipleMeasurementConfig = MeasurementConfig + + # current config should not have been setup yet + assert not connected_driver._connection._current_config + await connected_driver.initialize_measurement( + [450, 500, 600], mode=ABSMeasurementMode.MULTI + ) + + conf = cast( + AbsorbanceHidInterface.MultiMeasurementConfig, + connected_driver._connection._current_config, + ) + assert conf and conf.sample_wavelengths == [450, 500, 600] + mock_interface.abs96_initialize_multiple_measurement.assert_called_once_with( 1, conf ) - # setup up mock interface - MEASURE_RESULT = [0.1] * 96 - mock_interface.byonoy_abs96_single_measure.return_value = ( - MockErrorCode.BYONOY_ERROR_NO_ERROR, + # setup up mock interface with multiple readings + MEASURE_RESULT = [[0.1] * 96, [0.2] * 96, [0.3] * 96] + mock_interface.abs96_multiple_measure.return_value = ( + MockErrorCode.NO_ERROR, MEASURE_RESULT, ) - result = await connected_driver.get_single_measurement(450) - mock_interface.byonoy_abs96_single_measure.assert_called_once_with(1, conf) + result = await connected_driver.get_measurement() + mock_interface.abs96_multiple_measure.assert_called_once_with(1, conf) assert result == MEASURE_RESULT diff --git a/api/tests/opentrons/protocol_api/core/engine/test_absorbance_reader_core.py b/api/tests/opentrons/protocol_api/core/engine/test_absorbance_reader_core.py index 613ee3cfbe3..405d737d55b 100644 --- a/api/tests/opentrons/protocol_api/core/engine/test_absorbance_reader_core.py +++ b/api/tests/opentrons/protocol_api/core/engine/test_absorbance_reader_core.py @@ -62,32 +62,65 @@ def test_initialize( decoy: Decoy, mock_engine_client: EngineClient, subject: AbsorbanceReaderCore ) -> None: """It should set the sample wavelength with the engine client.""" - subject.initialize(wavelength=123) + subject.initialize("single", [123]) decoy.verify( mock_engine_client.execute_command( cmd.absorbance_reader.InitializeParams( moduleId="1234", - sampleWavelength=123, + measureMode="single", + sampleWavelengths=[123], + referenceWavelength=None, ), ), times=1, ) - assert subject._initialized_value == 123 + assert subject._initialized_value == [123] + + # Test reference wavelength + subject.initialize("single", [124], 450) + + decoy.verify( + mock_engine_client.execute_command( + cmd.absorbance_reader.InitializeParams( + moduleId="1234", + measureMode="single", + sampleWavelengths=[124], + referenceWavelength=450, + ), + ), + times=1, + ) + assert subject._initialized_value == [124] + + # Test initialize multi + subject.initialize("multi", [124, 125, 126]) + + decoy.verify( + mock_engine_client.execute_command( + cmd.absorbance_reader.InitializeParams( + moduleId="1234", + measureMode="multi", + sampleWavelengths=[124, 125, 126], + referenceWavelength=None, + ), + ), + times=1, + ) + assert subject._initialized_value == [124, 125, 126] def test_read( decoy: Decoy, mock_engine_client: EngineClient, subject: AbsorbanceReaderCore ) -> None: """It should call absorbance reader to read with the engine client.""" - subject._initialized_value = 123 + subject._initialized_value = [123] subject.read() decoy.verify( mock_engine_client.execute_command( cmd.absorbance_reader.ReadAbsorbanceParams( moduleId="1234", - sampleWavelength=123, ), ), times=1, diff --git a/api/tests/opentrons/protocol_api/core/engine/test_deck_conflict.py b/api/tests/opentrons/protocol_api/core/engine/test_deck_conflict.py index 147368e0734..9a46318c8b8 100644 --- a/api/tests/opentrons/protocol_api/core/engine/test_deck_conflict.py +++ b/api/tests/opentrons/protocol_api/core/engine/test_deck_conflict.py @@ -6,6 +6,7 @@ from opentrons_shared_data.labware.types import LabwareUri from opentrons_shared_data.robot.types import RobotType +from opentrons.hardware_control import CriticalPoint from opentrons.hardware_control.nozzle_manager import NozzleConfigurationType from opentrons.motion_planning import deck_conflict as wrapped_deck_conflict from opentrons.motion_planning import adjacent_slots_getters @@ -545,9 +546,21 @@ def test_deck_conflict_raises_for_bad_pipette_move( well_location=WellLocation(origin=WellOrigin.TOP, offset=WellOffset(z=10)), ) ).then_return(destination_well_point) + decoy.when( + mock_state_view.labware.get_should_center_column_on_target_well( + "destination-labware-id" + ) + ).then_return(False) + decoy.when( + mock_state_view.labware.get_should_center_pipette_on_target_well( + "destination-labware-id" + ) + ).then_return(False) decoy.when( mock_state_view.pipettes.get_pipette_bounds_at_specified_move_to_position( - pipette_id="pipette-id", destination_position=destination_well_point + pipette_id="pipette-id", + destination_position=destination_well_point, + critical_point=None, ) ).then_return(pipette_bounds) @@ -653,9 +666,17 @@ def test_deck_conflict_raises_for_collision_with_tc_lid( well_location=WellLocation(origin=WellOrigin.TOP, offset=WellOffset(z=10)), ) ).then_return(destination_well_point) + + decoy.when( + mock_state_view.labware.get_should_center_column_on_target_well( + "destination-labware-id" + ) + ).then_return(True) decoy.when( mock_state_view.pipettes.get_pipette_bounds_at_specified_move_to_position( - pipette_id="pipette-id", destination_position=destination_well_point + pipette_id="pipette-id", + destination_position=destination_well_point, + critical_point=CriticalPoint.Y_CENTER, ) ).then_return(pipette_bounds_at_destination) decoy.when(mock_state_view.pipettes.get_mount("pipette-id")).then_return( diff --git a/api/tests/opentrons/protocol_api_integration/test_pipette_movement_deck_conflicts.py b/api/tests/opentrons/protocol_api_integration/test_pipette_movement_deck_conflicts.py index 1d3388d3d97..ebaf5e49971 100644 --- a/api/tests/opentrons/protocol_api_integration/test_pipette_movement_deck_conflicts.py +++ b/api/tests/opentrons/protocol_api_integration/test_pipette_movement_deck_conflicts.py @@ -3,7 +3,7 @@ import pytest from opentrons import simulate -from opentrons.protocol_api import COLUMN, ALL, SINGLE +from opentrons.protocol_api import COLUMN, ALL, SINGLE, ROW from opentrons.protocol_api.core.engine.deck_conflict import ( PartialTipMovementNotAllowedError, ) @@ -226,3 +226,104 @@ def test_deck_conflicts_for_96_ch_a1_column_configuration() -> None: # No error NOW because of full config instrument.dispense(50, badly_placed_plate.wells_by_name()["A1"].bottom()) + + +@pytest.mark.ot3_only +def test_deck_conflicts_for_96_ch_and_reservoirs() -> None: + """It should raise errors for expected deck conflicts when moving to reservoirs. + + This test checks that the critical point of the pipette is taken into account, + specifically when it differs from the primary nozzle. + """ + protocol = simulate.get_protocol_api(version="2.20", robot_type="Flex") + instrument = protocol.load_instrument("flex_96channel_1000", mount="left") + # trash_labware = protocol.load_labware("opentrons_1_trash_3200ml_fixed", "A3") + # instrument.trash_container = trash_labware + + protocol.load_trash_bin("A3") + right_tiprack = protocol.load_labware("opentrons_flex_96_tiprack_50ul", "C3") + front_tiprack = protocol.load_labware("opentrons_flex_96_tiprack_50ul", "D2") + # Tall deck item in B3 + protocol.load_labware( + "opentrons_flex_96_tiprack_50ul", + "B3", + adapter="opentrons_flex_96_tiprack_adapter", + ) + # Tall deck item in B1 + protocol.load_labware( + "opentrons_flex_96_tiprack_50ul", + "B1", + adapter="opentrons_flex_96_tiprack_adapter", + ) + + # ############ RESERVOIRS ################ + # These labware should be to the east of tall labware to avoid any partial tip deck conflicts + reservoir_1_well = protocol.load_labware("nest_1_reservoir_195ml", "C2") + reservoir_12_well = protocol.load_labware("nest_12_reservoir_15ml", "B2") + + # ########### Use COLUMN A1 Config ############# + instrument.configure_nozzle_layout(style=COLUMN, start="A1") + + instrument.pick_up_tip(front_tiprack.wells_by_name()["A12"]) + + with pytest.raises( + PartialTipMovementNotAllowedError, match="collision with items in deck slot" + ): + instrument.aspirate(10, reservoir_1_well.wells()[0]) + + instrument.aspirate(25, reservoir_12_well.wells()[0]) + instrument.dispense(10, reservoir_12_well.wells()[1]) + + with pytest.raises( + PartialTipMovementNotAllowedError, match="collision with items in deck slot" + ): + instrument.dispense(15, reservoir_12_well.wells()[3]) + + instrument.drop_tip() + front_tiprack.reset() + + # ########### Use COLUMN A12 Config ############# + instrument.configure_nozzle_layout(style=COLUMN, start="A12") + + instrument.pick_up_tip(front_tiprack.wells_by_name()["A1"]) + instrument.aspirate(50, reservoir_1_well.wells()[0]) + with pytest.raises( + PartialTipMovementNotAllowedError, match="collision with items in deck slot" + ): + instrument.dispense(10, reservoir_12_well.wells()[8]) + + instrument.dispense(15, reservoir_12_well.wells()[11]) + instrument.dispense(10, reservoir_1_well.wells()[0]) + + instrument.drop_tip() + front_tiprack.reset() + + # ######## CHANGE CONFIG TO ROW H1 ######### + instrument.configure_nozzle_layout(style=ROW, start="H1", tip_racks=[front_tiprack]) + with pytest.raises( + PartialTipMovementNotAllowedError, match="collision with items in deck slot" + ): + instrument.pick_up_tip(right_tiprack.wells_by_name()["A1"]) + + instrument.pick_up_tip() + instrument.aspirate(25, reservoir_1_well.wells()[0]) + + instrument.drop_tip() + front_tiprack.reset() + + # ######## CHANGE CONFIG TO ROW A1 ######### + instrument.configure_nozzle_layout(style=ROW, start="A1", tip_racks=[front_tiprack]) + + with pytest.raises( + PartialTipMovementNotAllowedError, match="outside of robot bounds" + ): + instrument.pick_up_tip() + instrument.pick_up_tip(right_tiprack.wells_by_name()["H1"]) + + with pytest.raises( + PartialTipMovementNotAllowedError, match="collision with items in deck slot" + ): + instrument.aspirate(25, reservoir_1_well.wells()[0]) + + instrument.drop_tip() + front_tiprack.reset() diff --git a/api/tests/opentrons/protocol_engine/commands/test_liquid_probe.py b/api/tests/opentrons/protocol_engine/commands/test_liquid_probe.py index bd28916239b..6fb6ebc6935 100644 --- a/api/tests/opentrons/protocol_engine/commands/test_liquid_probe.py +++ b/api/tests/opentrons/protocol_engine/commands/test_liquid_probe.py @@ -4,6 +4,7 @@ from opentrons.protocol_engine.errors.exceptions import ( MustHomeError, + PipetteNotReadyToAspirateError, TipNotAttachedError, TipNotEmptyError, ) @@ -14,6 +15,7 @@ import pytest from opentrons.protocol_engine.commands.pipetting_common import LiquidNotFoundError +from opentrons.protocol_engine.state.state import StateView from opentrons.protocol_engine.state import update_types from opentrons.types import MountType, Point from opentrons.protocol_engine import WellLocation, WellOrigin, WellOffset, DeckPoint @@ -80,12 +82,14 @@ def result_type(types: tuple[object, object, EitherResultType]) -> EitherResultT @pytest.fixture def subject( implementation_type: EitherImplementationType, + state_view: StateView, movement: MovementHandler, pipetting: PipettingHandler, model_utils: ModelUtils, ) -> Union[LiquidProbeImplementation, TryLiquidProbeImplementation]: """Get the implementation subject.""" return implementation_type( + state_view=state_view, pipetting=pipetting, movement=movement, model_utils=model_utils, @@ -95,6 +99,7 @@ def subject( async def test_liquid_probe_implementation( decoy: Decoy, movement: MovementHandler, + state_view: StateView, pipetting: PipettingHandler, subject: EitherImplementation, params_type: EitherParamsType, @@ -110,7 +115,9 @@ async def test_liquid_probe_implementation( wellLocation=location, ) - decoy.when(pipetting.get_is_ready_to_aspirate(pipette_id="abc")).then_return(True) + decoy.when(state_view.pipettes.get_aspirated_volume(pipette_id="abc")).then_return( + 0 + ) decoy.when( await movement.move_to_well( @@ -148,6 +155,7 @@ async def test_liquid_probe_implementation( async def test_liquid_not_found_error( decoy: Decoy, + state_view: StateView, movement: MovementHandler, pipetting: PipettingHandler, subject: EitherImplementation, @@ -174,9 +182,7 @@ async def test_liquid_not_found_error( wellLocation=well_location, ) - decoy.when(pipetting.get_is_ready_to_aspirate(pipette_id=pipette_id)).then_return( - True - ) + decoy.when(state_view.pipettes.get_aspirated_volume(pipette_id)).then_return(0) decoy.when( await movement.move_to_well( @@ -230,11 +236,11 @@ async def test_liquid_not_found_error( async def test_liquid_probe_tip_checking( decoy: Decoy, - pipetting: PipettingHandler, + state_view: StateView, subject: EitherImplementation, params_type: EitherParamsType, ) -> None: - """It should return a TipNotAttached error if the hardware API indicates that.""" + """It should raise a TipNotAttached error if the state view indicates that.""" pipette_id = "pipette-id" labware_id = "labware-id" well_name = "well-name" @@ -249,18 +255,42 @@ async def test_liquid_probe_tip_checking( wellLocation=well_location, ) - decoy.when( - pipetting.get_is_ready_to_aspirate( - pipette_id=pipette_id, - ), - ).then_raise(TipNotAttachedError()) + decoy.when(state_view.pipettes.get_aspirated_volume(pipette_id)).then_raise( + TipNotAttachedError() + ) with pytest.raises(TipNotAttachedError): await subject.execute(data) +async def test_liquid_probe_plunger_preparedness_checking( + decoy: Decoy, + state_view: StateView, + subject: EitherImplementation, + params_type: EitherParamsType, +) -> None: + """It should raise a PipetteNotReadyToAspirate error if the state view indicates that.""" + pipette_id = "pipette-id" + labware_id = "labware-id" + well_name = "well-name" + well_location = WellLocation( + origin=WellOrigin.BOTTOM, offset=WellOffset(x=0, y=0, z=1) + ) + + data = params_type( + pipetteId=pipette_id, + labwareId=labware_id, + wellName=well_name, + wellLocation=well_location, + ) + + decoy.when(state_view.pipettes.get_aspirated_volume(pipette_id)).then_return(None) + with pytest.raises(PipetteNotReadyToAspirateError): + await subject.execute(data) + + async def test_liquid_probe_volume_checking( decoy: Decoy, - pipetting: PipettingHandler, + state_view: StateView, subject: EitherImplementation, params_type: EitherParamsType, ) -> None: @@ -278,15 +308,23 @@ async def test_liquid_probe_volume_checking( wellName=well_name, wellLocation=well_location, ) + decoy.when( - pipetting.get_is_empty(pipette_id=pipette_id), - ).then_return(False) + state_view.pipettes.get_aspirated_volume(pipette_id=pipette_id), + ).then_return(123) with pytest.raises(TipNotEmptyError): await subject.execute(data) + decoy.when( + state_view.pipettes.get_aspirated_volume(pipette_id=pipette_id), + ).then_return(None) + with pytest.raises(PipetteNotReadyToAspirateError): + await subject.execute(data) + async def test_liquid_probe_location_checking( decoy: Decoy, + state_view: StateView, movement: MovementHandler, subject: EitherImplementation, params_type: EitherParamsType, @@ -305,6 +343,7 @@ async def test_liquid_probe_location_checking( wellName=well_name, wellLocation=well_location, ) + decoy.when(state_view.pipettes.get_aspirated_volume(pipette_id)).then_return(0) decoy.when( await movement.check_for_valid_position( mount=MountType.LEFT, diff --git a/api/tests/opentrons/protocol_engine/commands/test_load_labware.py b/api/tests/opentrons/protocol_engine/commands/test_load_labware.py index c9bf96c12c2..85cb7794d76 100644 --- a/api/tests/opentrons/protocol_engine/commands/test_load_labware.py +++ b/api/tests/opentrons/protocol_engine/commands/test_load_labware.py @@ -1,5 +1,10 @@ """Test load labware commands.""" import inspect +from typing import Optional +from opentrons.protocol_engine.state.update_types import ( + LoadedLabwareUpdate, + StateUpdate, +) import pytest from decoy import Decoy @@ -37,11 +42,13 @@ def patch_mock_labware_validation( monkeypatch.setattr(labware_validation, name, decoy.mock(func=func)) +@pytest.mark.parametrize("display_name", [("My custom display name"), (None)]) async def test_load_labware_implementation( decoy: Decoy, well_plate_def: LabwareDefinition, equipment: EquipmentHandler, state_view: StateView, + display_name: Optional[str], ) -> None: """A LoadLabware command should have an execution implementation.""" subject = LoadLabwareImplementation(equipment=equipment, state_view=state_view) @@ -51,7 +58,7 @@ async def test_load_labware_implementation( loadName="some-load-name", namespace="opentrons-test", version=1, - displayName="My custom display name", + displayName=display_name, ) decoy.when( @@ -88,6 +95,15 @@ async def test_load_labware_implementation( offsetId="labware-offset-id", ), private=None, + state_update=StateUpdate( + loaded_labware=LoadedLabwareUpdate( + labware_id="labware-id", + definition=well_plate_def, + offset_id="labware-offset-id", + new_location=DeckSlotLocation(slotName=DeckSlotName.SLOT_4), + display_name=display_name, + ) + ), ) @@ -156,7 +172,6 @@ async def test_load_labware_on_labware( ).then_return(True) result = await subject.execute(data) - assert result == SuccessData( public=LoadLabwareResult( labwareId="labware-id", @@ -164,6 +179,15 @@ async def test_load_labware_on_labware( offsetId="labware-offset-id", ), private=None, + state_update=StateUpdate( + loaded_labware=LoadedLabwareUpdate( + labware_id="labware-id", + definition=well_plate_def, + offset_id="labware-offset-id", + new_location=OnLabwareLocation(labwareId="another-labware-id"), + display_name="My custom display name", + ) + ), ) decoy.verify( diff --git a/api/tests/opentrons/protocol_engine/commands/test_move_labware.py b/api/tests/opentrons/protocol_engine/commands/test_move_labware.py index b80da9d902d..cc6a75bc020 100644 --- a/api/tests/opentrons/protocol_engine/commands/test_move_labware.py +++ b/api/tests/opentrons/protocol_engine/commands/test_move_labware.py @@ -107,6 +107,13 @@ async def test_manual_move_labware_implementation( offsetId="wowzers-a-new-offset-id", ), private=None, + state_update=update_types.StateUpdate( + labware_location=update_types.LabwareLocationUpdate( + labware_id="my-cool-labware-id", + offset_id="wowzers-a-new-offset-id", + new_location=DeckSlotLocation(slotName=DeckSlotName.SLOT_5), + ) + ), ) @@ -173,6 +180,13 @@ async def test_move_labware_implementation_on_labware( offsetId="wowzers-a-new-offset-id", ), private=None, + state_update=update_types.StateUpdate( + labware_location=update_types.LabwareLocationUpdate( + labware_id="my-cool-labware-id", + offset_id="wowzers-a-new-offset-id", + new_location=OnLabwareLocation(labwareId="my-even-cooler-labware-id"), + ) + ), ) @@ -260,7 +274,14 @@ async def test_gripper_move_labware_implementation( offsetId="wowzers-a-new-offset-id", ), private=None, - state_update=update_types.StateUpdate(pipette_location=update_types.CLEAR), + state_update=update_types.StateUpdate( + pipette_location=update_types.CLEAR, + labware_location=update_types.LabwareLocationUpdate( + labware_id="my-cool-labware-id", + new_location=new_location, + offset_id="wowzers-a-new-offset-id", + ), + ), ) @@ -409,7 +430,14 @@ async def test_gripper_move_to_waste_chute_implementation( offsetId="wowzers-a-new-offset-id", ), private=None, - state_update=update_types.StateUpdate(pipette_location=update_types.CLEAR), + state_update=update_types.StateUpdate( + pipette_location=update_types.CLEAR, + labware_location=update_types.LabwareLocationUpdate( + labware_id="my-cool-labware-id", + new_location=new_location, + offset_id="wowzers-a-new-offset-id", + ), + ), ) diff --git a/api/tests/opentrons/protocol_engine/commands/test_reload_labware.py b/api/tests/opentrons/protocol_engine/commands/test_reload_labware.py index 51b3f24b753..c79727c9a31 100644 --- a/api/tests/opentrons/protocol_engine/commands/test_reload_labware.py +++ b/api/tests/opentrons/protocol_engine/commands/test_reload_labware.py @@ -1,5 +1,9 @@ """Test load labware commands.""" import inspect +from opentrons.protocol_engine.state.update_types import ( + LabwareLocationUpdate, + StateUpdate, +) import pytest from decoy import Decoy @@ -63,6 +67,13 @@ async def test_reload_labware_implementation( offsetId="labware-offset-id", ), private=None, + state_update=StateUpdate( + labware_location=LabwareLocationUpdate( + labware_id="my-labware-id", + new_location=DeckSlotLocation(slotName=DeckSlotName.SLOT_4), + offset_id="labware-offset-id", + ) + ), ) diff --git a/api/tests/opentrons/protocol_engine/execution/test_equipment_handler.py b/api/tests/opentrons/protocol_engine/execution/test_equipment_handler.py index 370b76a16dd..b7a020c2d35 100644 --- a/api/tests/opentrons/protocol_engine/execution/test_equipment_handler.py +++ b/api/tests/opentrons/protocol_engine/execution/test_equipment_handler.py @@ -53,9 +53,9 @@ ) from opentrons.protocol_engine.execution.equipment import ( EquipmentHandler, - LoadedLabwareData, LoadedPipetteData, LoadedModuleData, + LoadedLabwareData, ) from ..pipette_fixtures import get_default_nozzle_map diff --git a/api/tests/opentrons/protocol_engine/execution/test_tip_handler.py b/api/tests/opentrons/protocol_engine/execution/test_tip_handler.py index 560675b8190..5abeb7308b6 100644 --- a/api/tests/opentrons/protocol_engine/execution/test_tip_handler.py +++ b/api/tests/opentrons/protocol_engine/execution/test_tip_handler.py @@ -3,10 +3,10 @@ from decoy import Decoy from mock import AsyncMock, patch -from typing import Dict, ContextManager, Optional +from typing import Dict, ContextManager, Optional, OrderedDict from contextlib import nullcontext as does_not_raise -from opentrons.types import Mount, MountType +from opentrons.types import Mount, MountType, Point from opentrons.hardware_control import API as HardwareAPI from opentrons.hardware_control.types import TipStateType from opentrons.hardware_control.protocols.types import OT2RobotType, FlexRobotType @@ -25,6 +25,8 @@ VirtualTipHandler, create_tip_handler, ) +from opentrons.hardware_control.nozzle_manager import NozzleMap +from opentrons_shared_data.pipette.pipette_definition import ValidNozzleMaps @pytest.fixture @@ -53,6 +55,17 @@ def tip_rack_definition() -> LabwareDefinition: return LabwareDefinition.construct(namespace="test", version=42) # type: ignore[call-arg] +MOCK_MAP = NozzleMap.build( + physical_nozzles=OrderedDict({"A1": Point(0, 0, 0)}), + physical_rows=OrderedDict({"A": ["A1"]}), + physical_columns=OrderedDict({"1": ["A1"]}), + starting_nozzle="A1", + back_left_nozzle="A1", + front_right_nozzle="A1", + valid_nozzle_maps=ValidNozzleMaps(maps={"Full": ["A1"]}), +) + + async def test_create_tip_handler( decoy: Decoy, mock_state_view: StateView, @@ -102,6 +115,9 @@ async def test_flex_pick_up_tip_state( decoy.when(mock_state_view.pipettes.get_mount("pipette-id")).then_return( MountType.LEFT ) + decoy.when(mock_state_view.pipettes.state.nozzle_configuration_by_id).then_return( + {"pipette-id": MOCK_MAP} + ) decoy.when( mock_state_view.geometry.get_nominal_tip_geometry( pipette_id="pipette-id", @@ -170,6 +186,10 @@ async def test_pick_up_tip( MountType.LEFT ) + decoy.when(mock_state_view.pipettes.state.nozzle_configuration_by_id).then_return( + {"pipette-id": MOCK_MAP} + ) + decoy.when( mock_state_view.geometry.get_nominal_tip_geometry( pipette_id="pipette-id", @@ -224,6 +244,9 @@ async def test_drop_tip( decoy.when(mock_state_view.pipettes.get_mount("pipette-id")).then_return( MountType.RIGHT ) + decoy.when(mock_state_view.pipettes.state.nozzle_configuration_by_id).then_return( + {"pipette-id": MOCK_MAP} + ) await subject.drop_tip(pipette_id="pipette-id", home_after=True) @@ -498,6 +521,11 @@ async def test_verify_tip_presence_on_ot3( decoy.when(mock_state_view.pipettes.get_mount("pipette-id")).then_return( MountType.LEFT ) + + decoy.when( + mock_state_view.pipettes.state.nozzle_configuration_by_id + ).then_return({"pipette-id": MOCK_MAP}) + await subject.verify_tip_presence("pipette-id", expected, None) decoy.verify( diff --git a/api/tests/opentrons/protocol_engine/state/command_fixtures.py b/api/tests/opentrons/protocol_engine/state/command_fixtures.py index df7fb4dca9a..53025629920 100644 --- a/api/tests/opentrons/protocol_engine/state/command_fixtures.py +++ b/api/tests/opentrons/protocol_engine/state/command_fixtures.py @@ -113,6 +113,22 @@ def create_succeeded_command( ) +def create_comment_command(command_id: str = "command-id") -> cmd.Comment: + """Create a completed LoadLabware command.""" + params = cmd.CommentParams(message="hello world!") + + result = cmd.CommentResult() + + return cmd.Comment( + id=command_id, + key="command-key", + status=cmd.CommandStatus.SUCCEEDED, + createdAt=datetime.now(), + params=params, + result=result, + ) + + def create_load_labware_command( labware_id: str, location: LabwareLocation, diff --git a/api/tests/opentrons/protocol_engine/state/test_geometry_view.py b/api/tests/opentrons/protocol_engine/state/test_geometry_view.py index 1854d08523a..6bbd13c5e25 100644 --- a/api/tests/opentrons/protocol_engine/state/test_geometry_view.py +++ b/api/tests/opentrons/protocol_engine/state/test_geometry_view.py @@ -1,6 +1,10 @@ """Test state getters for retrieving geometry views of state.""" import inspect import json +from opentrons.protocol_engine.state.update_types import ( + LoadedLabwareUpdate, + StateUpdate, +) import pytest from math import isclose from decoy import Decoy @@ -2528,6 +2532,15 @@ def test_get_offset_location_deck_slot( ), ), private_result=None, + state_update=StateUpdate( + loaded_labware=LoadedLabwareUpdate( + labware_id="labware-id-1", + definition=nice_labware_definition, + offset_id=None, + new_location=DeckSlotLocation(slotName=DeckSlotName.SLOT_C2), + display_name=None, + ) + ), ) labware_store.handle_action(action) offset_location = subject.get_offset_location("labware-id-1") @@ -2584,7 +2597,17 @@ def test_get_offset_location_module( ), ), private_result=None, + state_update=StateUpdate( + loaded_labware=LoadedLabwareUpdate( + labware_id="labware-id-1", + definition=nice_labware_definition, + offset_id=None, + new_location=ModuleLocation(moduleId="module-id-1"), + display_name=None, + ) + ), ) + module_store.handle_action(load_module) labware_store.handle_action(load_labware) offset_location = subject.get_offset_location("labware-id-1") @@ -2643,6 +2666,15 @@ def test_get_offset_location_module_with_adapter( ), ), private_result=None, + state_update=StateUpdate( + loaded_labware=LoadedLabwareUpdate( + labware_id="adapter-id-1", + definition=nice_adapter_definition, + offset_id=None, + new_location=ModuleLocation(moduleId="module-id-1"), + display_name=None, + ) + ), ) load_labware = SucceedCommandAction( command=LoadLabware( @@ -2663,6 +2695,15 @@ def test_get_offset_location_module_with_adapter( ), ), private_result=None, + state_update=StateUpdate( + loaded_labware=LoadedLabwareUpdate( + labware_id="labware-id-1", + definition=nice_labware_definition, + offset_id=None, + new_location=OnLabwareLocation(labwareId="adapter-id-1"), + display_name=None, + ) + ), ) module_store.handle_action(load_module) labware_store.handle_action(load_adapter) @@ -2703,6 +2744,15 @@ def test_get_offset_fails_with_off_deck_labware( ), ), private_result=None, + state_update=StateUpdate( + loaded_labware=LoadedLabwareUpdate( + labware_id="labware-id-1", + definition=nice_labware_definition, + offset_id=None, + new_location=OFF_DECK_LOCATION, + display_name=None, + ) + ), ) labware_store.handle_action(action) offset_location = subject.get_offset_location("labware-id-1") diff --git a/api/tests/opentrons/protocol_engine/state/test_labware_store.py b/api/tests/opentrons/protocol_engine/state/test_labware_store.py index cb651fc37a7..68c7e86c5ff 100644 --- a/api/tests/opentrons/protocol_engine/state/test_labware_store.py +++ b/api/tests/opentrons/protocol_engine/state/test_labware_store.py @@ -1,4 +1,6 @@ """Labware state store tests.""" +from typing import Optional +from opentrons.protocol_engine.state import update_types import pytest from datetime import datetime @@ -16,7 +18,6 @@ DeckSlotLocation, LoadedLabware, OFF_DECK_LOCATION, - LabwareMovementStrategy, ) from opentrons.protocol_engine.actions import ( AddLabwareOffsetAction, @@ -26,9 +27,7 @@ from opentrons.protocol_engine.state.labware import LabwareStore, LabwareState from .command_fixtures import ( - create_load_labware_command, - create_move_labware_command, - create_reload_labware_command, + create_comment_command, ) @@ -85,9 +84,14 @@ def test_handles_add_labware_offset( assert subject.state.labware_offsets_by_id == {"offset-id": resolved_offset} +@pytest.mark.parametrize( + "display_name, offset_id", [("display-name", "offset-id"), (None, None)] +) def test_handles_load_labware( subject: LabwareStore, well_plate_def: LabwareDefinition, + display_name: Optional[str], + offset_id: Optional[str], ) -> None: """It should add the labware data to the state.""" offset_request = LabwareOffsetCreate( @@ -96,13 +100,7 @@ def test_handles_load_labware( vector=LabwareOffsetVector(x=1, y=2, z=3), ) - command = create_load_labware_command( - location=DeckSlotLocation(slotName=DeckSlotName.SLOT_1), - labware_id="test-labware-id", - definition=well_plate_def, - offset_id="offset-id", - display_name="display-name", - ) + command = create_comment_command() expected_definition_uri = uri_from_details( load_name=well_plate_def.parameters.loadName, @@ -115,8 +113,8 @@ def test_handles_load_labware( loadName=well_plate_def.parameters.loadName, definitionUri=expected_definition_uri, location=DeckSlotLocation(slotName=DeckSlotName.SLOT_1), - offsetId="offset-id", - displayName="display-name", + offsetId=offset_id, + displayName=display_name, ) subject.handle_action( @@ -126,7 +124,21 @@ def test_handles_load_labware( created_at=datetime(year=2021, month=1, day=2), ) ) - subject.handle_action(SucceedCommandAction(private_result=None, command=command)) + subject.handle_action( + SucceedCommandAction( + private_result=None, + command=command, + state_update=update_types.StateUpdate( + loaded_labware=update_types.LoadedLabwareUpdate( + labware_id="test-labware-id", + definition=well_plate_def, + offset_id=offset_id, + display_name=display_name, + new_location=DeckSlotLocation(slotName=DeckSlotName.SLOT_1), + ), + ), + ) + ) assert subject.state.labware_by_id["test-labware-id"] == expected_labware_data @@ -138,16 +150,22 @@ def test_handles_reload_labware( well_plate_def: LabwareDefinition, ) -> None: """It should override labware data in the state.""" - load_labware = create_load_labware_command( - location=DeckSlotLocation(slotName=DeckSlotName.SLOT_A1), - labware_id="test-labware-id", - definition=well_plate_def, - display_name="display-name", - offset_id=None, - ) + command = create_comment_command() subject.handle_action( - SucceedCommandAction(private_result=None, command=load_labware) + SucceedCommandAction( + private_result=None, + command=command, + state_update=update_types.StateUpdate( + loaded_labware=update_types.LoadedLabwareUpdate( + labware_id="test-labware-id", + definition=well_plate_def, + offset_id=None, + display_name="display-name", + new_location=DeckSlotLocation(slotName=DeckSlotName.SLOT_1), + ), + ), + ) ) expected_definition_uri = uri_from_details( load_name=well_plate_def.parameters.loadName, @@ -171,19 +189,28 @@ def test_handles_reload_labware( created_at=datetime(year=2021, month=1, day=2), ) ) - reload_labware = create_reload_labware_command( - labware_id="test-labware-id", - offset_id="offset-id", + comment_command_2 = create_comment_command( + command_id="comment-id-1", ) subject.handle_action( - SucceedCommandAction(private_result=None, command=reload_labware) + SucceedCommandAction( + private_result=None, + command=comment_command_2, + state_update=update_types.StateUpdate( + labware_location=update_types.LabwareLocationUpdate( + new_location=DeckSlotLocation(slotName=DeckSlotName.SLOT_1), + offset_id="offset-id", + labware_id="test-labware-id", + ) + ), + ) ) expected_labware_data = LoadedLabware( id="test-labware-id", loadName=well_plate_def.parameters.loadName, definitionUri=expected_definition_uri, - location=DeckSlotLocation(slotName=DeckSlotName.SLOT_A1), + location=DeckSlotLocation(slotName=DeckSlotName.SLOT_1), offsetId="offset-id", displayName="display-name", ) @@ -212,13 +239,7 @@ def test_handles_move_labware( well_plate_def: LabwareDefinition, ) -> None: """It should update labware state with new location & offset.""" - load_labware_command = create_load_labware_command( - location=DeckSlotLocation(slotName=DeckSlotName.SLOT_1), - labware_id="my-labware-id", - definition=well_plate_def, - offset_id="old-offset-id", - display_name="display-name", - ) + comment_command = create_comment_command() offset_request = LabwareOffsetCreate( definitionUri="offset-definition-uri", location=LabwareOffsetLocation(slotName=DeckSlotName.SLOT_1), @@ -232,21 +253,40 @@ def test_handles_move_labware( ) ) subject.handle_action( - SucceedCommandAction(private_result=None, command=load_labware_command) + SucceedCommandAction( + private_result=None, + command=comment_command, + state_update=update_types.StateUpdate( + loaded_labware=update_types.LoadedLabwareUpdate( + labware_id="my-labware-id", + definition=well_plate_def, + offset_id=None, + display_name="display-name", + new_location=DeckSlotLocation(slotName=DeckSlotName.SLOT_1), + ), + ), + ) ) - move_command = create_move_labware_command( - labware_id="my-labware-id", - new_location=DeckSlotLocation(slotName=DeckSlotName.SLOT_4), - offset_id="my-new-offset", - strategy=LabwareMovementStrategy.MANUAL_MOVE_WITH_PAUSE, + comment_2 = create_comment_command( + command_id="my-command-id", ) subject.handle_action( - SucceedCommandAction(private_result=None, command=move_command) + SucceedCommandAction( + private_result=None, + command=comment_2, + state_update=update_types.StateUpdate( + labware_location=update_types.LabwareLocationUpdate( + labware_id="my-labware-id", + offset_id="my-new-offset", + new_location=DeckSlotLocation(slotName=DeckSlotName.SLOT_1), + ), + ), + ) ) assert subject.state.labware_by_id["my-labware-id"].location == DeckSlotLocation( - slotName=DeckSlotName.SLOT_4 + slotName=DeckSlotName.SLOT_1 ) assert subject.state.labware_by_id["my-labware-id"].offsetId == "my-new-offset" @@ -256,13 +296,7 @@ def test_handles_move_labware_off_deck( well_plate_def: LabwareDefinition, ) -> None: """It should update labware state with new location & offset.""" - load_labware_command = create_load_labware_command( - location=DeckSlotLocation(slotName=DeckSlotName.SLOT_1), - labware_id="my-labware-id", - definition=well_plate_def, - offset_id="old-offset-id", - display_name="display-name", - ) + comment_command = create_comment_command() offset_request = LabwareOffsetCreate( definitionUri="offset-definition-uri", location=LabwareOffsetLocation(slotName=DeckSlotName.SLOT_1), @@ -276,16 +310,36 @@ def test_handles_move_labware_off_deck( ) ) subject.handle_action( - SucceedCommandAction(private_result=None, command=load_labware_command) + SucceedCommandAction( + private_result=None, + command=comment_command, + state_update=update_types.StateUpdate( + loaded_labware=update_types.LoadedLabwareUpdate( + labware_id="my-labware-id", + definition=well_plate_def, + offset_id=None, + display_name="display-name", + new_location=DeckSlotLocation(slotName=DeckSlotName.SLOT_1), + ), + ), + ) ) - move_labware_off_deck_cmd = create_move_labware_command( - labware_id="my-labware-id", - new_location=OFF_DECK_LOCATION, - strategy=LabwareMovementStrategy.MANUAL_MOVE_WITH_PAUSE, + comment_2 = create_comment_command( + command_id="my-command-id", ) subject.handle_action( - SucceedCommandAction(private_result=None, command=move_labware_off_deck_cmd) + SucceedCommandAction( + private_result=None, + command=comment_2, + state_update=update_types.StateUpdate( + labware_location=update_types.LabwareLocationUpdate( + labware_id="my-labware-id", + new_location=OFF_DECK_LOCATION, + offset_id=None, + ) + ), + ) ) assert subject.state.labware_by_id["my-labware-id"].location == OFF_DECK_LOCATION assert subject.state.labware_by_id["my-labware-id"].offsetId is None diff --git a/api/tests/opentrons/protocol_engine/state/test_pipette_view.py b/api/tests/opentrons/protocol_engine/state/test_pipette_view.py index e8823c3c6ad..27bee5f1d15 100644 --- a/api/tests/opentrons/protocol_engine/state/test_pipette_view.py +++ b/api/tests/opentrons/protocol_engine/state/test_pipette_view.py @@ -9,6 +9,7 @@ from opentrons_shared_data.pipette.pipette_definition import ValidNozzleMaps from opentrons.config.defaults_ot2 import Z_RETRACT_DISTANCE +from opentrons.hardware_control import CriticalPoint from opentrons.types import MountType, Mount as HwMount, Point from opentrons.hardware_control.dev_types import PipetteDict from opentrons.protocol_engine import errors @@ -586,8 +587,9 @@ class _PipetteSpecs(NamedTuple): tip_length: float bounding_box_offsets: PipetteBoundingBoxOffsets nozzle_map: NozzleMap + critical_point: Optional[CriticalPoint] destination_position: Point - nozzle_bounds_result: Tuple[Point, Point, Point, Point] + pipette_bounds_result: Tuple[Point, Point, Point, Point] _pipette_spec_cases = [ @@ -609,8 +611,9 @@ class _PipetteSpecs(NamedTuple): front_right_nozzle="H1", valid_nozzle_maps=ValidNozzleMaps(maps={"Full": EIGHT_CHANNEL_COLS["1"]}), ), + critical_point=None, destination_position=Point(100, 200, 300), - nozzle_bounds_result=( + pipette_bounds_result=( ( Point(x=100.0, y=200.0, z=342.0), Point(x=100.0, y=137.0, z=342.0), @@ -637,8 +640,9 @@ class _PipetteSpecs(NamedTuple): front_right_nozzle="H1", valid_nozzle_maps=ValidNozzleMaps(maps={"H1": ["H1"]}), ), + critical_point=None, destination_position=Point(100, 200, 300), - nozzle_bounds_result=( + pipette_bounds_result=( ( Point(x=100.0, y=263.0, z=342.0), Point(x=100.0, y=200.0, z=342.0), @@ -647,6 +651,68 @@ class _PipetteSpecs(NamedTuple): ) ), ), + _PipetteSpecs( + # 8-channel P300, full configuration. Critical point of XY_CENTER + tip_length=42, + bounding_box_offsets=PipetteBoundingBoxOffsets( + back_left_corner=Point(0.0, 31.5, 35.52), + front_right_corner=Point(0.0, -31.5, 35.52), + front_left_corner=Point(0.0, -31.5, 35.52), + back_right_corner=Point(0.0, 31.5, 35.52), + ), + nozzle_map=NozzleMap.build( + physical_nozzles=EIGHT_CHANNEL_MAP, + physical_rows=EIGHT_CHANNEL_ROWS, + physical_columns=EIGHT_CHANNEL_COLS, + starting_nozzle="A1", + back_left_nozzle="A1", + front_right_nozzle="H1", + valid_nozzle_maps=ValidNozzleMaps(maps={"Full": EIGHT_CHANNEL_COLS["1"]}), + ), + critical_point=CriticalPoint.XY_CENTER, + destination_position=Point(100, 200, 300), + pipette_bounds_result=( + ( + Point(x=100.0, y=231.5, z=342.0), + Point(x=100.0, y=168.5, z=342.0), + Point(x=100.0, y=231.5, z=342.0), + Point(x=100.0, y=168.5, z=342.0), + ) + ), + ), + _PipetteSpecs( + # 8-channel P300, Partial A1-E1 configuration. Critical point of XY_CENTER + tip_length=42, + bounding_box_offsets=PipetteBoundingBoxOffsets( + back_left_corner=Point(0.0, 31.5, 35.52), + front_right_corner=Point(0.0, -31.5, 35.52), + front_left_corner=Point(0.0, -31.5, 35.52), + back_right_corner=Point(0.0, 31.5, 35.52), + ), + nozzle_map=NozzleMap.build( + physical_nozzles=EIGHT_CHANNEL_MAP, + physical_rows=EIGHT_CHANNEL_ROWS, + physical_columns=EIGHT_CHANNEL_COLS, + starting_nozzle="H1", + back_left_nozzle="E1", + front_right_nozzle="H1", + valid_nozzle_maps=ValidNozzleMaps( + maps={ + "H1toE1": ["E1", "F1", "G1", "H1"], + } + ), + ), + critical_point=CriticalPoint.XY_CENTER, + destination_position=Point(100, 200, 300), + pipette_bounds_result=( + ( + Point(x=100.0, y=249.5, z=342.0), + Point(x=100.0, y=186.5, z=342.0), + Point(x=100.0, y=249.5, z=342.0), + Point(x=100.0, y=186.5, z=342.0), + ) + ), + ), _PipetteSpecs( # 96-channel P1000, full configuration tip_length=42, @@ -681,8 +747,9 @@ class _PipetteSpecs(NamedTuple): } ), ), + critical_point=None, destination_position=Point(100, 200, 300), - nozzle_bounds_result=( + pipette_bounds_result=( ( Point(x=100.0, y=200.0, z=342.0), Point(x=199.0, y=137.0, z=342.0), @@ -709,8 +776,9 @@ class _PipetteSpecs(NamedTuple): front_right_nozzle="H1", valid_nozzle_maps=ValidNozzleMaps(maps={"Column1": NINETY_SIX_COLS["1"]}), ), + critical_point=None, destination_position=Point(100, 200, 300), - nozzle_bounds_result=( + pipette_bounds_result=( Point(100, 200, 342), Point(199, 137, 342), Point(199, 200, 342), @@ -735,8 +803,9 @@ class _PipetteSpecs(NamedTuple): front_right_nozzle="H12", valid_nozzle_maps=ValidNozzleMaps(maps={"Column12": NINETY_SIX_COLS["12"]}), ), + critical_point=None, destination_position=Point(100, 200, 300), - nozzle_bounds_result=( + pipette_bounds_result=( Point(1, 200, 342), Point(100, 137, 342), Point(100, 200, 342), @@ -761,14 +830,96 @@ class _PipetteSpecs(NamedTuple): front_right_nozzle="A12", valid_nozzle_maps=ValidNozzleMaps(maps={"RowA": NINETY_SIX_ROWS["A"]}), ), + critical_point=None, destination_position=Point(100, 200, 300), - nozzle_bounds_result=( + pipette_bounds_result=( Point(100, 200, 342), Point(199, 137, 342), Point(199, 200, 342), Point(100, 137, 342), ), ), + _PipetteSpecs( + # 96-channel P1000, ROW configuration. Critical point of XY_CENTER + tip_length=42, + bounding_box_offsets=PipetteBoundingBoxOffsets( + back_left_corner=Point(-36.0, -25.5, -259.15), + front_right_corner=Point(63.0, -88.5, -259.15), + front_left_corner=Point(-36.0, -88.5, -259.15), + back_right_corner=Point(63.0, -25.5, -259.15), + ), + nozzle_map=NozzleMap.build( + physical_nozzles=NINETY_SIX_MAP, + physical_rows=NINETY_SIX_ROWS, + physical_columns=NINETY_SIX_COLS, + starting_nozzle="A1", + back_left_nozzle="A1", + front_right_nozzle="A12", + valid_nozzle_maps=ValidNozzleMaps(maps={"RowA": NINETY_SIX_ROWS["A"]}), + ), + critical_point=CriticalPoint.XY_CENTER, + destination_position=Point(100, 200, 300), + pipette_bounds_result=( + Point(x=50.5, y=200, z=342), + Point(x=149.5, y=137, z=342), + Point(x=149.5, y=200, z=342), + Point(x=50.5, y=137, z=342), + ), + ), + _PipetteSpecs( + # 96-channel P1000, A12 COLUMN configuration. Critical point of Y_CENTER + tip_length=42, + bounding_box_offsets=PipetteBoundingBoxOffsets( + back_left_corner=Point(-36.0, -25.5, -259.15), + front_right_corner=Point(63.0, -88.5, -259.15), + front_left_corner=Point(-36.0, -88.5, -259.15), + back_right_corner=Point(63.0, -25.5, -259.15), + ), + nozzle_map=NozzleMap.build( + physical_nozzles=NINETY_SIX_MAP, + physical_rows=NINETY_SIX_ROWS, + physical_columns=NINETY_SIX_COLS, + starting_nozzle="A12", + back_left_nozzle="A12", + front_right_nozzle="H12", + valid_nozzle_maps=ValidNozzleMaps(maps={"Column12": NINETY_SIX_COLS["12"]}), + ), + critical_point=CriticalPoint.Y_CENTER, + destination_position=Point(100, 200, 300), + pipette_bounds_result=( + Point(1, 231.5, 342), + Point(100, 168.5, 342), + Point(100, 231.5, 342), + Point(1, 168.5, 342), + ), + ), + _PipetteSpecs( + # 96-channel P1000, A1 COLUMN configuration. Critical point of XY_CENTER + tip_length=42, + bounding_box_offsets=PipetteBoundingBoxOffsets( + back_left_corner=Point(-36.0, -25.5, -259.15), + front_right_corner=Point(63.0, -88.5, -259.15), + front_left_corner=Point(-36.0, -88.5, -259.15), + back_right_corner=Point(63.0, -25.5, -259.15), + ), + nozzle_map=NozzleMap.build( + physical_nozzles=NINETY_SIX_MAP, + physical_rows=NINETY_SIX_ROWS, + physical_columns=NINETY_SIX_COLS, + starting_nozzle="A1", + back_left_nozzle="A1", + front_right_nozzle="H1", + valid_nozzle_maps=ValidNozzleMaps(maps={"Column1": NINETY_SIX_COLS["1"]}), + ), + critical_point=CriticalPoint.XY_CENTER, + destination_position=Point(100, 200, 300), + pipette_bounds_result=( + Point(100, 231.5, 342), + Point(199, 168.5, 342), + Point(199, 231.5, 342), + Point(100, 168.5, 342), + ), + ), ] @@ -776,12 +927,13 @@ class _PipetteSpecs(NamedTuple): argnames=_PipetteSpecs._fields, argvalues=_pipette_spec_cases, ) -def test_get_nozzle_bounds_at_location( +def test_get_pipette_bounds_at_location( tip_length: float, bounding_box_offsets: PipetteBoundingBoxOffsets, nozzle_map: NozzleMap, destination_position: Point, - nozzle_bounds_result: Tuple[Point, Point, Point, Point], + critical_point: Optional[CriticalPoint], + pipette_bounds_result: Tuple[Point, Point, Point, Point], ) -> None: """It should get the pipette's nozzle's bounds at the given location.""" subject = get_pipette_view( @@ -810,7 +962,9 @@ def test_get_nozzle_bounds_at_location( ) assert ( subject.get_pipette_bounds_at_specified_move_to_position( - pipette_id="pipette-id", destination_position=destination_position + pipette_id="pipette-id", + destination_position=destination_position, + critical_point=critical_point, ) - == nozzle_bounds_result + == pipette_bounds_result ) diff --git a/api/tests/opentrons/protocol_runner/test_json_translator.py b/api/tests/opentrons/protocol_runner/test_json_translator.py index d7c82ca7475..62181880dbc 100644 --- a/api/tests/opentrons/protocol_runner/test_json_translator.py +++ b/api/tests/opentrons/protocol_runner/test_json_translator.py @@ -706,7 +706,7 @@ def _load_labware_definition_data() -> LabwareDefinition: ], bottomShape=SphericalSegment( shape="spherical", - radius_of_curvature=6, + radiusOfCurvature=6, depth=10, ), ) diff --git a/api/tests/opentrons/protocol_runner/test_legacy_command_mapper.py b/api/tests/opentrons/protocol_runner/test_legacy_command_mapper.py index ed171280d17..b58032d6cba 100644 --- a/api/tests/opentrons/protocol_runner/test_legacy_command_mapper.py +++ b/api/tests/opentrons/protocol_runner/test_legacy_command_mapper.py @@ -3,6 +3,10 @@ from datetime import datetime from typing import cast +from opentrons.protocol_engine.state.update_types import ( + LoadedLabwareUpdate, + StateUpdate, +) import pytest from decoy import matchers, Decoy @@ -316,6 +320,15 @@ def test_map_labware_load(minimal_labware_def: LabwareDefinition) -> None: notes=[], ), private_result=None, + state_update=StateUpdate( + loaded_labware=LoadedLabwareUpdate( + labware_id="labware-0", + definition=matchers.Anything(), + offset_id="labware-offset-id-123", + new_location=DeckSlotLocation(slotName=DeckSlotName.SLOT_1), + display_name="My special labware", + ) + ), ) result_queue, result_run, result_succeed = LegacyCommandMapper().map_equipment_load( input @@ -500,6 +513,15 @@ def test_map_module_labware_load(minimal_labware_def: LabwareDefinition) -> None notes=[], ), private_result=None, + state_update=StateUpdate( + loaded_labware=LoadedLabwareUpdate( + labware_id="labware-0", + definition=matchers.Anything(), + offset_id="labware-offset-id-123", + new_location=ModuleLocation(moduleId="module-123"), + display_name="My very special module labware", + ) + ), ) subject = LegacyCommandMapper() diff --git a/api/tests/opentrons/protocols/geometry/test_frustum_helpers.py b/api/tests/opentrons/protocols/geometry/test_frustum_helpers.py new file mode 100644 index 00000000000..0bf74aae5b2 --- /dev/null +++ b/api/tests/opentrons/protocols/geometry/test_frustum_helpers.py @@ -0,0 +1,265 @@ +import pytest +from math import pi, isclose +from typing import Any, List + +from opentrons_shared_data.labware.types import ( + RectangularBoundedSection, + CircularBoundedSection, + SphericalSegment, +) +from opentrons.protocol_engine.state.frustum_helpers import ( + cross_section_area_rectangular, + cross_section_area_circular, + reject_unacceptable_heights, + get_boundary_pairs, + circular_frustum_polynomial_roots, + rectangular_frustum_polynomial_roots, + volume_from_height_rectangular, + volume_from_height_circular, + volume_from_height_spherical, + height_from_volume_circular, + height_from_volume_rectangular, + height_from_volume_spherical, +) +from opentrons.protocol_engine.errors.exceptions import InvalidLiquidHeightFound + + +def fake_frusta() -> List[List[Any]]: + """A bunch of weird fake well shapes.""" + frusta = [] + frusta.append( + [ + RectangularBoundedSection( + shape="rectangular", xDimension=9.0, yDimension=10.0, topHeight=10.0 + ), + RectangularBoundedSection( + shape="rectangular", xDimension=8.0, yDimension=9.0, topHeight=5.0 + ), + CircularBoundedSection(shape="circular", diameter=23.0, topHeight=1.0), + SphericalSegment(shape="spherical", radiusOfCurvature=4.0, depth=1.0), + ] + ) + frusta.append( + [ + RectangularBoundedSection( + shape="rectangular", xDimension=8.0, yDimension=70.0, topHeight=3.5 + ), + RectangularBoundedSection( + shape="rectangular", xDimension=8.0, yDimension=75.0, topHeight=2.0 + ), + RectangularBoundedSection( + shape="rectangular", xDimension=8.0, yDimension=80.0, topHeight=1.0 + ), + RectangularBoundedSection( + shape="rectangular", xDimension=8.0, yDimension=90.0, topHeight=0.0 + ), + ] + ) + frusta.append( + [ + CircularBoundedSection(shape="circular", diameter=23.0, topHeight=7.5), + CircularBoundedSection(shape="circular", diameter=11.5, topHeight=5.0), + CircularBoundedSection(shape="circular", diameter=23.0, topHeight=2.5), + CircularBoundedSection(shape="circular", diameter=11.5, topHeight=0.0), + ] + ) + frusta.append( + [ + CircularBoundedSection(shape="circular", diameter=4.0, topHeight=3.0), + CircularBoundedSection(shape="circular", diameter=5.0, topHeight=2.0), + SphericalSegment(shape="spherical", radiusOfCurvature=3.5, depth=2.0), + ] + ) + frusta.append( + [SphericalSegment(shape="spherical", radiusOfCurvature=4.0, depth=3.0)] + ) + frusta.append( + [ + RectangularBoundedSection( + shape="rectangular", xDimension=27.0, yDimension=36.0, topHeight=3.5 + ), + RectangularBoundedSection( + shape="rectangular", xDimension=36.0, yDimension=26.0, topHeight=1.5 + ), + SphericalSegment(shape="spherical", radiusOfCurvature=4.0, depth=1.5), + ] + ) + return frusta + + +@pytest.mark.parametrize( + ["max_height", "potential_heights", "expected_heights"], + [ + (34, [complex(4, 5), complex(5, 0), 35, 34, 33, 10, 0], [5, 34, 33, 10, 0]), + (2934, [complex(4, 5), complex(5, 0)], [5]), + (100, [-99, -1, complex(99.99, 0), 101], [99.99]), + (2, [0, -1, complex(-1.5, 0)], [0]), + (8, [complex(7, 1), -0.01], []), + ], +) +def test_reject_unacceptable_heights( + max_height: float, potential_heights: List[Any], expected_heights: List[float] +) -> None: + """Make sure we reject all mathematical solutions that are physically not possible.""" + if len(expected_heights) != 1: + with pytest.raises(InvalidLiquidHeightFound): + reject_unacceptable_heights( + max_height=max_height, potential_heights=potential_heights + ) + else: + found_heights = reject_unacceptable_heights( + max_height=max_height, potential_heights=potential_heights + ) + assert found_heights == expected_heights[0] + + +@pytest.mark.parametrize("diameter", [2, 5, 8, 356, 1000]) +def test_cross_section_area_circular(diameter: float) -> None: + """Test circular area calculation.""" + expected_area = pi * (diameter / 2) ** 2 + assert cross_section_area_circular(diameter) == expected_area + + +@pytest.mark.parametrize( + ["x_dimension", "y_dimension"], [(1, 38402), (234, 983), (94857, 40), (234, 999)] +) +def test_cross_section_area_rectangular(x_dimension: float, y_dimension: float) -> None: + """Test rectangular area calculation.""" + expected_area = x_dimension * y_dimension + assert ( + cross_section_area_rectangular(x_dimension=x_dimension, y_dimension=y_dimension) + == expected_area + ) + + +@pytest.mark.parametrize("well", fake_frusta()) +def test_get_cross_section_boundaries(well: List[List[Any]]) -> None: + """Make sure get_cross_section_boundaries returns the expected list indices.""" + i = 0 + for f, next_f in get_boundary_pairs(well): + assert f == well[i] + assert next_f == well[i + 1] + i += 1 + + +@pytest.mark.parametrize("well", fake_frusta()) +def test_volume_and_height_circular(well: List[Any]) -> None: + """Test both volume and height calculations for circular frusta.""" + if well[-1]["shape"] == "spherical": + return + total_height = well[0]["topHeight"] + for f, next_f in get_boundary_pairs(well): + if f["shape"] == next_f["shape"] == "circular": + top_radius = next_f["diameter"] / 2 + bottom_radius = f["diameter"] / 2 + a = pi * ((top_radius - bottom_radius) ** 2) / (3 * total_height**2) + b = pi * bottom_radius * (top_radius - bottom_radius) / total_height + c = pi * bottom_radius**2 + assert circular_frustum_polynomial_roots( + top_radius=top_radius, + bottom_radius=bottom_radius, + total_frustum_height=total_height, + ) == (a, b, c) + # test volume within a bunch of arbitrary heights + for target_height in range(round(total_height)): + expected_volume = ( + a * (target_height**3) + + b * (target_height**2) + + c * target_height + ) + found_volume = volume_from_height_circular( + target_height=target_height, + total_frustum_height=total_height, + bottom_radius=bottom_radius, + top_radius=top_radius, + ) + assert found_volume == expected_volume + # test going backwards to get height back + found_height = height_from_volume_circular( + volume=found_volume, + total_frustum_height=total_height, + bottom_radius=bottom_radius, + top_radius=top_radius, + ) + assert isclose(found_height, target_height) + + +@pytest.mark.parametrize("well", fake_frusta()) +def test_volume_and_height_rectangular(well: List[Any]) -> None: + """Test both volume and height calculations for rectangular frusta.""" + if well[-1]["shape"] == "spherical": + return + total_height = well[0]["topHeight"] + for f, next_f in get_boundary_pairs(well): + if f["shape"] == next_f["shape"] == "rectangular": + top_length = next_f["yDimension"] + top_width = next_f["xDimension"] + bottom_length = f["yDimension"] + bottom_width = f["xDimension"] + a = ( + (top_length - bottom_length) + * (top_width - bottom_width) + / (3 * total_height**2) + ) + b = ( + (bottom_length * (top_width - bottom_width)) + + (bottom_width * (top_length - bottom_length)) + ) / (2 * total_height) + c = bottom_length * bottom_width + assert rectangular_frustum_polynomial_roots( + top_length=top_length, + bottom_length=bottom_length, + top_width=top_width, + bottom_width=bottom_width, + total_frustum_height=total_height, + ) == (a, b, c) + # test volume within a bunch of arbitrary heights + for target_height in range(round(total_height)): + expected_volume = ( + a * (target_height**3) + + b * (target_height**2) + + c * target_height + ) + found_volume = volume_from_height_rectangular( + target_height=target_height, + total_frustum_height=total_height, + bottom_length=bottom_length, + bottom_width=bottom_width, + top_length=top_length, + top_width=top_width, + ) + assert found_volume == expected_volume + # test going backwards to get height back + found_height = height_from_volume_rectangular( + volume=found_volume, + total_frustum_height=total_height, + bottom_length=bottom_length, + bottom_width=bottom_width, + top_length=top_length, + top_width=top_width, + ) + assert isclose(found_height, target_height) + + +@pytest.mark.parametrize("well", fake_frusta()) +def test_volume_and_height_spherical(well: List[Any]) -> None: + """Test both volume and height calculations for spherical segments.""" + if well[0]["shape"] == "spherical": + for target_height in range(round(well[0]["depth"])): + expected_volume = ( + (1 / 3) + * pi + * (target_height**2) + * (3 * well[0]["radiusOfCurvature"] - target_height) + ) + found_volume = volume_from_height_spherical( + target_height=target_height, + radius_of_curvature=well[0]["radiusOfCurvature"], + ) + assert found_volume == expected_volume + found_height = height_from_volume_spherical( + volume=found_volume, + radius_of_curvature=well[0]["radiusOfCurvature"], + total_frustum_height=well[0]["depth"], + ) + assert isclose(found_height, target_height) diff --git a/app-shell/build/release-notes.md b/app-shell/build/release-notes.md index 5ab9009bebc..80b36253b85 100644 --- a/app-shell/build/release-notes.md +++ b/app-shell/build/release-notes.md @@ -27,8 +27,9 @@ Welcome to the v8.0.0 release of the Opentrons App! ### Known Issues -- Labware offsets can't be applied to protocols that require selecting a CSV file as a runtime parameter value. Write the protocol in such a way that it passes analysis with or without the CSV file, or run Labware Position Check after confirming parameter values. +- Stored labware offsets can't be applied to protocols that require selecting a CSV file as a runtime parameter value. Write the protocol in such a way that it passes analysis with or without the CSV file, or run Labware Position Check after confirming parameter values. - Error recovery can't perform partial tip pickup, because it doesn't account for the pipette nozzle configuration of 8- and 96-channel pipettes. If a recovery step requires partial tip pickup, cancel the protocol instead. +- Downloading robot logs via USB may take up to 2 minutes on macOS, and may fail entirely on Windows. Use an Ethernet or Wi-Fi connection to download logs if needed. --- diff --git a/app/src/App/DesktopApp.tsx b/app/src/App/DesktopApp.tsx index b59f76804a6..2a6edade61d 100644 --- a/app/src/App/DesktopApp.tsx +++ b/app/src/App/DesktopApp.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, Fragment } from 'react' import { Navigate, Route, Routes, useMatch } from 'react-router-dom' import { ErrorBoundary } from 'react-error-boundary' import { I18nextProvider } from 'react-i18next' @@ -46,7 +46,7 @@ export const DesktopApp = (): JSX.Element => { const [ isEmergencyStopModalDismissed, setIsEmergencyStopModalDismissed, - ] = React.useState(false) + ] = useState(false) const desktopRoutes: RouteProps[] = [ { @@ -124,7 +124,7 @@ export const DesktopApp = (): JSX.Element => { + { - + } path={path} /> diff --git a/app/src/App/DesktopAppFallback.tsx b/app/src/App/DesktopAppFallback.tsx index a1bb6528f45..1bf2645e8b1 100644 --- a/app/src/App/DesktopAppFallback.tsx +++ b/app/src/App/DesktopAppFallback.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useDispatch } from 'react-redux' import { useNavigate } from 'react-router-dom' import { useTranslation } from 'react-i18next' diff --git a/app/src/App/ODDTopLevelRedirects/index.tsx b/app/src/App/ODDTopLevelRedirects/index.tsx index 2bda4c52f94..fa0692bd1c4 100644 --- a/app/src/App/ODDTopLevelRedirects/index.tsx +++ b/app/src/App/ODDTopLevelRedirects/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Navigate, Route, Routes } from 'react-router-dom' import { useCurrentRunId } from '/app/resources/runs' diff --git a/app/src/App/OnDeviceDisplayApp.tsx b/app/src/App/OnDeviceDisplayApp.tsx index 812b05185a3..396cfdb4484 100644 --- a/app/src/App/OnDeviceDisplayApp.tsx +++ b/app/src/App/OnDeviceDisplayApp.tsx @@ -1,6 +1,6 @@ -import * as React from 'react' +import { useEffect, useState, useCallback } from 'react' import { useDispatch, useSelector } from 'react-redux' -import { Navigate, Route, Routes } from 'react-router-dom' +import { Routes, Route, Navigate, useLocation } from 'react-router-dom' import { css } from 'styled-components' import { ErrorBoundary } from 'react-error-boundary' @@ -163,7 +163,7 @@ export const OnDeviceDisplayApp = (): JSX.Element => { const dispatch = useDispatch() const isIdle = useIdle(sleepTime, options) - React.useEffect(() => { + useEffect(() => { if (isIdle) { dispatch(updateBrightness(TURN_OFF_BACKLIGHT)) } else { @@ -220,11 +220,19 @@ const getTargetPath = (unfinishedUnboxingFlowRoute: string | null): string => { // split to a separate function because scrollRef rerenders on every route change // this avoids rerendering parent providers as well export function OnDeviceDisplayAppRoutes(): JSX.Element { - const [currentNode, setCurrentNode] = React.useState(null) - const scrollRef = React.useCallback((node: HTMLElement | null) => { + const [currentNode, setCurrentNode] = useState(null) + const scrollRef = useCallback((node: HTMLElement | null) => { setCurrentNode(node) }, []) const isScrolling = useScrolling(currentNode) + const location = useLocation() + useEffect(() => { + currentNode?.scrollTo({ + top: 0, + left: 0, + behavior: 'auto', + }) + }, [location.pathname]) const { unfinishedUnboxingFlowRoute } = useSelector( getOnDeviceDisplaySettings diff --git a/app/src/App/OnDeviceDisplayAppFallback.tsx b/app/src/App/OnDeviceDisplayAppFallback.tsx index 1e507463445..6077db2d4da 100644 --- a/app/src/App/OnDeviceDisplayAppFallback.tsx +++ b/app/src/App/OnDeviceDisplayAppFallback.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useEffect } from 'react' import { useDispatch, useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' @@ -47,7 +47,7 @@ export function OnDeviceDisplayAppFallback({ } // immediately report to robot logs that something fatal happened - React.useEffect(() => { + useEffect(() => { dispatch(sendLog(`ODD app encountered a fatal error: ${error.message}`)) }, []) diff --git a/app/src/App/__mocks__/portal.tsx b/app/src/App/__mocks__/portal.tsx index 67fd57af2de..f80b1deb44e 100644 --- a/app/src/App/__mocks__/portal.tsx +++ b/app/src/App/__mocks__/portal.tsx @@ -1,5 +1,5 @@ // mock portal for enzyme tests -import * as React from 'react' +import type * as React from 'react' interface Props { children: React.ReactNode diff --git a/app/src/App/__tests__/App.test.tsx b/app/src/App/__tests__/App.test.tsx index cd7a08e1c2c..0f5faf53894 100644 --- a/app/src/App/__tests__/App.test.tsx +++ b/app/src/App/__tests__/App.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, describe, beforeEach, afterEach, expect, it } from 'vitest' import { when } from 'vitest-when' import { screen } from '@testing-library/react' diff --git a/app/src/App/__tests__/DesktopApp.test.tsx b/app/src/App/__tests__/DesktopApp.test.tsx index e43cae4a097..1daf5c7d226 100644 --- a/app/src/App/__tests__/DesktopApp.test.tsx +++ b/app/src/App/__tests__/DesktopApp.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { screen } from '@testing-library/react' import { when } from 'vitest-when' diff --git a/app/src/App/__tests__/Navbar.test.tsx b/app/src/App/__tests__/Navbar.test.tsx index db7eedc744b..508323c739c 100644 --- a/app/src/App/__tests__/Navbar.test.tsx +++ b/app/src/App/__tests__/Navbar.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { describe, it } from 'vitest' import { screen, render } from '@testing-library/react' import { MemoryRouter } from 'react-router-dom' diff --git a/app/src/App/__tests__/OnDeviceDisplayApp.test.tsx b/app/src/App/__tests__/OnDeviceDisplayApp.test.tsx index 5243db5310a..fae54eb2fed 100644 --- a/app/src/App/__tests__/OnDeviceDisplayApp.test.tsx +++ b/app/src/App/__tests__/OnDeviceDisplayApp.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { vi, describe, beforeEach, afterEach, expect, it } from 'vitest' import { MemoryRouter } from 'react-router-dom' diff --git a/app/src/App/__tests__/OnDeviceDisplayAppFallback.test.tsx b/app/src/App/__tests__/OnDeviceDisplayAppFallback.test.tsx index d04a0f8df1c..864e714694a 100644 --- a/app/src/App/__tests__/OnDeviceDisplayAppFallback.test.tsx +++ b/app/src/App/__tests__/OnDeviceDisplayAppFallback.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, describe, beforeEach, it, expect } from 'vitest' import { fireEvent, screen } from '@testing-library/react' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/App/__tests__/hooks.test.tsx b/app/src/App/__tests__/hooks.test.tsx index 9ce1f5e9160..5b3f315049b 100644 --- a/app/src/App/__tests__/hooks.test.tsx +++ b/app/src/App/__tests__/hooks.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, describe, beforeEach, afterEach, expect, it } from 'vitest' import { renderHook } from '@testing-library/react' import { createStore } from 'redux' diff --git a/app/src/App/hooks.ts b/app/src/App/hooks.ts index dd28b35e964..d01082d8dc1 100644 --- a/app/src/App/hooks.ts +++ b/app/src/App/hooks.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useCallback, useRef, useEffect } from 'react' import difference from 'lodash/difference' import { useTranslation } from 'react-i18next' import { useQueryClient } from 'react-query' @@ -23,7 +23,7 @@ const PROTOCOL_IDS_RECHECK_INTERVAL_MS = 3000 export function useSoftwareUpdatePoll(): void { const dispatch = useDispatch() - const checkAppUpdate = React.useCallback(() => dispatch(checkShellUpdate()), [ + const checkAppUpdate = useCallback(() => dispatch(checkShellUpdate()), [ dispatch, ]) useInterval(checkAppUpdate, UPDATE_RECHECK_INTERVAL_MS) @@ -41,8 +41,8 @@ export function useProtocolReceiptToast(): void { true ) const protocolIds = protocolIdsQuery.data?.data ?? [] - const protocolIdsRef = React.useRef(protocolIds) - const hasRefetched = React.useRef(true) + const protocolIdsRef = useRef(protocolIds) + const hasRefetched = useRef(true) const { createLiveCommand } = useCreateLiveCommandMutation() const animationCommand: SetStatusBarCreateCommand = { commandType: 'setStatusBar', @@ -53,7 +53,7 @@ export function useProtocolReceiptToast(): void { hasRefetched.current = false } - React.useEffect(() => { + useEffect(() => { const newProtocolIds = difference(protocolIds, protocolIdsRef.current) if (!hasRefetched.current && newProtocolIds.length > 0) { Promise.all( diff --git a/app/src/App/index.tsx b/app/src/App/index.tsx index 2e94c63d6f7..f0ba1de0304 100644 --- a/app/src/App/index.tsx +++ b/app/src/App/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useSelector } from 'react-redux' import { Flex, POSITION_FIXED, DIRECTION_ROW } from '@opentrons/components' diff --git a/app/src/App/portal.tsx b/app/src/App/portal.tsx index 346d5842d81..d9e17199f56 100644 --- a/app/src/App/portal.tsx +++ b/app/src/App/portal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Box } from '@opentrons/components' export const TOP_PORTAL_ID = '__otAppTopPortalRoot' diff --git a/app/src/DesignTokens/BorderRadius/BorderRadius.stories.tsx b/app/src/DesignTokens/BorderRadius/BorderRadius.stories.tsx index 677e26b0c77..29e655c3a11 100644 --- a/app/src/DesignTokens/BorderRadius/BorderRadius.stories.tsx +++ b/app/src/DesignTokens/BorderRadius/BorderRadius.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { ALIGN_FLEX_START, BORDERS, diff --git a/app/src/DesignTokens/Colors/Colors.stories.tsx b/app/src/DesignTokens/Colors/Colors.stories.tsx index 60bbdff3945..98601ccac19 100644 --- a/app/src/DesignTokens/Colors/Colors.stories.tsx +++ b/app/src/DesignTokens/Colors/Colors.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { ALIGN_FLEX_START, BORDERS, diff --git a/app/src/DesignTokens/Spacing/Spacing.stories.tsx b/app/src/DesignTokens/Spacing/Spacing.stories.tsx index 5a0d0d4da81..3b12ae833db 100644 --- a/app/src/DesignTokens/Spacing/Spacing.stories.tsx +++ b/app/src/DesignTokens/Spacing/Spacing.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { ALIGN_FLEX_START, Box, diff --git a/app/src/DesignTokens/Typography/Typography.stories.tsx b/app/src/DesignTokens/Typography/Typography.stories.tsx index ac95fa3a369..787d558bf52 100644 --- a/app/src/DesignTokens/Typography/Typography.stories.tsx +++ b/app/src/DesignTokens/Typography/Typography.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import type { FlattenSimpleInterpolation } from 'styled-components' import { diff --git a/app/src/LocalizationProvider.tsx b/app/src/LocalizationProvider.tsx index 7fbb30d4774..1cd676d2095 100644 --- a/app/src/LocalizationProvider.tsx +++ b/app/src/LocalizationProvider.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { I18nextProvider } from 'react-i18next' import reduce from 'lodash/reduce' diff --git a/app/src/__testing-utils__/renderWithProviders.tsx b/app/src/__testing-utils__/renderWithProviders.tsx index 65a2e01855e..11e3ba16d9b 100644 --- a/app/src/__testing-utils__/renderWithProviders.tsx +++ b/app/src/__testing-utils__/renderWithProviders.tsx @@ -1,6 +1,6 @@ // render using targetted component using @testing-library/react // with wrapping providers for i18next and redux -import * as React from 'react' +import type * as React from 'react' import { QueryClient, QueryClientProvider } from 'react-query' import { I18nextProvider } from 'react-i18next' import { Provider } from 'react-redux' diff --git a/app/src/assets/localization/en/error_recovery.json b/app/src/assets/localization/en/error_recovery.json index 605c2b55527..f9e9253a8a6 100644 --- a/app/src/assets/localization/en/error_recovery.json +++ b/app/src/assets/localization/en/error_recovery.json @@ -11,6 +11,7 @@ "change_location": "Change location", "change_tip_pickup_location": "Change tip pick-up location", "choose_a_recovery_action": "Choose a recovery action", + "close_door_to_resume": "Close robot door to resume", "close_the_robot_door": "Close the robot door, and then resume the recovery action.", "confirm": "Confirm", "continue": "Continue", @@ -21,9 +22,10 @@ "error_on_robot": "Error on {{robot}}", "failed_dispense_step_not_completed": "The failed dispense step will not be completed. The run will continue from the next step with the attached tips.Close the robot door before proceeding.", "failed_step": "Failed step", - "first_take_any_necessary_actions": "First, take any necessary actions to prepare the robot to retry the failed step.Then, close the robot door before proceeding.", "go_back": "Go back", "homing_pipette_dangerous": "Homing the {{mount}} pipette with liquid in the tips may damage it. You must remove all tips before using the pipette again.", + "if_issue_persists_overpressure": " If the issue persists, cancel the run and make the necessary changes to the protocol", + "if_issue_persists_tip_not_detected": " If the issue persists, cancel the run and initiate Labware Position Check", "if_tips_are_attached": "If tips are attached, you can choose to blow out any aspirated liquid and drop tips before the run is terminated.", "ignore_all_errors_of_this_type": "Ignore all errors of this type", "ignore_error_and_skip": "Ignore error and skip to next step", @@ -36,7 +38,6 @@ "next_try_another_action": "Next, you can try another recovery action or cancel the run.", "no_liquid_detected": "No liquid detected", "overpressure_is_usually_caused": "Overpressure is usually caused by a tip contacting labware, a clog, or moving viscous liquid too quickly", - "if_issue_persists": " If the issue persists, cancel the run and make the necessary changes to the protocol", "pick_up_tips": "Pick up tips", "pipette_overpressure": "Pipette overpressure", "preserve_aspirated_liquid": "First, do you need to preserve aspirated liquid?", @@ -75,9 +76,12 @@ "stand_back_resuming": "Stand back, resuming current step", "stand_back_retrying": "Stand back, retrying failed step", "stand_back_skipping_to_next_step": "Stand back, skipping to next step", + "take_necessary_actions": "First, take any necessary actions to prepare the robot to retry the failed step.Then, close the robot door before proceeding.", + "take_necessary_actions_failed_pickup": "First, take any necessary actions to prepare the robot to retry the failed tip pickup.Then, close the robot door before proceeding.", "terminate_remote_activity": "Terminate remote activity", "tip_drop_failed": "Tip drop failed", "tip_not_detected": "Tip not detected", + "tip_presence_errors_are_caused": "Tip presence errors are usually caused by improperly placed labware or inaccurate labware offsets", "view_error_details": "View error details", "view_recovery_options": "View recovery options", "you_can_still_drop_tips": "You can still drop the attached tips before proceeding to tip selection." diff --git a/app/src/assets/localization/en/protocol_command_text.json b/app/src/assets/localization/en/protocol_command_text.json index 379a047cda8..17d60a8f967 100644 --- a/app/src/assets/localization/en/protocol_command_text.json +++ b/app/src/assets/localization/en/protocol_command_text.json @@ -8,7 +8,7 @@ "closing_tc_lid": "Closing Thermocycler lid", "comment": "Comment", "configure_for_volume": "Configure {{pipette}} to aspirate {{volume}} µL", - "configure_nozzle_layout": "Configure {{pipette}} to use {{amount}} nozzles", + "configure_nozzle_layout": "Configure {{pipette}} to use {{layout}}", "confirm_and_resume": "Confirm and resume", "deactivate_hs_shake": "Deactivating shaker", "deactivate_temperature_module": "Deactivating Temperature Module", diff --git a/app/src/assets/localization/en/quick_transfer.json b/app/src/assets/localization/en/quick_transfer.json index 0cb31912964..32efac281bc 100644 --- a/app/src/assets/localization/en/quick_transfer.json +++ b/app/src/assets/localization/en/quick_transfer.json @@ -96,7 +96,7 @@ "pipette_path": "Pipette path", "pipette_path_multi_aspirate": "Multi-aspirate", "pipette_path_multi_dispense": "Multi-dispense", - "pipette_path_multi_dispense_volume_blowout": "Multi-dispense, {{volume}} disposal volume, blowout {{blowOutLocation}}", + "pipette_path_multi_dispense_volume_blowout": "Multi-dispense, {{volume}} µL disposal volume, blowout {{blowOutLocation}}", "pipette_path_single": "Single transfers", "pre_wet_tip": "Pre-wet tip", "quick_transfer": "Quick transfer", diff --git a/app/src/atoms/InlineNotification/InlineNotification.stories.tsx b/app/src/atoms/InlineNotification/InlineNotification.stories.tsx index b993fd13205..ff4e4de6999 100644 --- a/app/src/atoms/InlineNotification/InlineNotification.stories.tsx +++ b/app/src/atoms/InlineNotification/InlineNotification.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { VIEWPORT } from '@opentrons/components' import { InlineNotification } from '.' import type { Story, Meta } from '@storybook/react' diff --git a/app/src/atoms/InlineNotification/__tests__/InlineNotification.test.tsx b/app/src/atoms/InlineNotification/__tests__/InlineNotification.test.tsx index a1bdf9b33c2..0e91433e7b2 100644 --- a/app/src/atoms/InlineNotification/__tests__/InlineNotification.test.tsx +++ b/app/src/atoms/InlineNotification/__tests__/InlineNotification.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, beforeEach, expect } from 'vitest' import { screen, fireEvent } from '@testing-library/react' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/atoms/InlineNotification/index.tsx b/app/src/atoms/InlineNotification/index.tsx index 1605abd990d..5b5bf21aafa 100644 --- a/app/src/atoms/InlineNotification/index.tsx +++ b/app/src/atoms/InlineNotification/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { ALIGN_CENTER, diff --git a/app/src/atoms/InstrumentContainer/InstrumentContainer.stories.tsx b/app/src/atoms/InstrumentContainer/InstrumentContainer.stories.tsx index 680b104c6b8..d5daeb2d6d7 100644 --- a/app/src/atoms/InstrumentContainer/InstrumentContainer.stories.tsx +++ b/app/src/atoms/InstrumentContainer/InstrumentContainer.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { InstrumentContainer as InstrumentContainerComponent } from './index' diff --git a/app/src/atoms/InstrumentContainer/__tests__/InstrumentContainer.test.tsx b/app/src/atoms/InstrumentContainer/__tests__/InstrumentContainer.test.tsx index d7a6a7d41fd..e5fa872ab54 100644 --- a/app/src/atoms/InstrumentContainer/__tests__/InstrumentContainer.test.tsx +++ b/app/src/atoms/InstrumentContainer/__tests__/InstrumentContainer.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it } from 'vitest' import { screen } from '@testing-library/react' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/atoms/InstrumentContainer/index.tsx b/app/src/atoms/InstrumentContainer/index.tsx index 3efb742fa76..475dfed3b23 100644 --- a/app/src/atoms/InstrumentContainer/index.tsx +++ b/app/src/atoms/InstrumentContainer/index.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - import { BORDERS, COLORS, diff --git a/app/src/atoms/Link/ExternalLink.stories.tsx b/app/src/atoms/Link/ExternalLink.stories.tsx index 8f664d257f5..d5d2dccaee9 100644 --- a/app/src/atoms/Link/ExternalLink.stories.tsx +++ b/app/src/atoms/Link/ExternalLink.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { COLORS, Flex, SPACING } from '@opentrons/components' import { ExternalLink as ExternalLinkComponent } from './ExternalLink' diff --git a/app/src/atoms/Link/ExternalLink.tsx b/app/src/atoms/Link/ExternalLink.tsx index e35e3515277..5d24a06fdb4 100644 --- a/app/src/atoms/Link/ExternalLink.tsx +++ b/app/src/atoms/Link/ExternalLink.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Link, Icon, TYPOGRAPHY, SPACING } from '@opentrons/components' import type { LinkProps } from '@opentrons/components' diff --git a/app/src/atoms/Link/__tests__/ExternalLink.test.tsx b/app/src/atoms/Link/__tests__/ExternalLink.test.tsx index 81c9e8cc6c2..e245541c514 100644 --- a/app/src/atoms/Link/__tests__/ExternalLink.test.tsx +++ b/app/src/atoms/Link/__tests__/ExternalLink.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, beforeEach } from 'vitest' import { screen } from '@testing-library/react' import '@testing-library/jest-dom/vitest' diff --git a/app/src/atoms/ProgressBar/__tests__/ProgressBar.test.tsx b/app/src/atoms/ProgressBar/__tests__/ProgressBar.test.tsx index daaf7c70417..6d5b3d3fa40 100644 --- a/app/src/atoms/ProgressBar/__tests__/ProgressBar.test.tsx +++ b/app/src/atoms/ProgressBar/__tests__/ProgressBar.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { screen } from '@testing-library/react' diff --git a/app/src/atoms/ProgressBar/index.tsx b/app/src/atoms/ProgressBar/index.tsx index 01054384094..5852dadf2b5 100644 --- a/app/src/atoms/ProgressBar/index.tsx +++ b/app/src/atoms/ProgressBar/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { COLORS, Box } from '@opentrons/components' diff --git a/app/src/atoms/SelectField/Select.stories.tsx b/app/src/atoms/SelectField/Select.stories.tsx index 2df1ac462fa..461bab59b22 100644 --- a/app/src/atoms/SelectField/Select.stories.tsx +++ b/app/src/atoms/SelectField/Select.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Select } from './Select' diff --git a/app/src/atoms/SelectField/Select.tsx b/app/src/atoms/SelectField/Select.tsx index 2f1a6ef8879..110fbbbf857 100644 --- a/app/src/atoms/SelectField/Select.tsx +++ b/app/src/atoms/SelectField/Select.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import ReactSelect, { components } from 'react-select' import { diff --git a/app/src/atoms/SelectField/index.tsx b/app/src/atoms/SelectField/index.tsx index 36c425cbeae..50deed28266 100644 --- a/app/src/atoms/SelectField/index.tsx +++ b/app/src/atoms/SelectField/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import find from 'lodash/find' import { Select } from './Select' import { diff --git a/app/src/atoms/Skeleton/Skeleton.stories.tsx b/app/src/atoms/Skeleton/Skeleton.stories.tsx index f3c4ab231c4..c91679c1a3a 100644 --- a/app/src/atoms/Skeleton/Skeleton.stories.tsx +++ b/app/src/atoms/Skeleton/Skeleton.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Flex, DIRECTION_COLUMN, diff --git a/app/src/atoms/Skeleton/__tests__/Skeleton.test.tsx b/app/src/atoms/Skeleton/__tests__/Skeleton.test.tsx index 9e37a85622c..b03bd72e98d 100644 --- a/app/src/atoms/Skeleton/__tests__/Skeleton.test.tsx +++ b/app/src/atoms/Skeleton/__tests__/Skeleton.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect } from 'vitest' import '@testing-library/jest-dom/vitest' import { screen } from '@testing-library/react' diff --git a/app/src/atoms/Skeleton/index.tsx b/app/src/atoms/Skeleton/index.tsx index 556a5653a77..3c9ccb09a7b 100644 --- a/app/src/atoms/Skeleton/index.tsx +++ b/app/src/atoms/Skeleton/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { BORDERS, Box, COLORS } from '@opentrons/components' diff --git a/app/src/atoms/SleepScreen/__tests__/SleepScreen.test.tsx b/app/src/atoms/SleepScreen/__tests__/SleepScreen.test.tsx index a1862f37db0..79356e879e0 100644 --- a/app/src/atoms/SleepScreen/__tests__/SleepScreen.test.tsx +++ b/app/src/atoms/SleepScreen/__tests__/SleepScreen.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { describe, it, expect } from 'vitest' import '@testing-library/jest-dom/vitest' import { screen } from '@testing-library/react' diff --git a/app/src/atoms/SleepScreen/index.tsx b/app/src/atoms/SleepScreen/index.tsx index 2600f9d2de9..64ff2ff7cbe 100644 --- a/app/src/atoms/SleepScreen/index.tsx +++ b/app/src/atoms/SleepScreen/index.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - import { Flex, COLORS } from '@opentrons/components' export function SleepScreen(): JSX.Element { diff --git a/app/src/atoms/Slideout/MultiSlideout.tsx b/app/src/atoms/Slideout/MultiSlideout.tsx index 73054a10a45..65865edec53 100644 --- a/app/src/atoms/Slideout/MultiSlideout.tsx +++ b/app/src/atoms/Slideout/MultiSlideout.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Slideout } from './index' import type { MultiSlideoutSpecs, SlideoutProps } from './index' diff --git a/app/src/atoms/Slideout/Slideout.stories.tsx b/app/src/atoms/Slideout/Slideout.stories.tsx index 6d1b828573c..9cee5939d15 100644 --- a/app/src/atoms/Slideout/Slideout.stories.tsx +++ b/app/src/atoms/Slideout/Slideout.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { Fragment } from 'react' import { action } from '@storybook/addon-actions' import { COLORS, @@ -24,7 +24,7 @@ export default meta type Story = StoryObj const Children = ( - + - + ) export const Slideout: Story = { diff --git a/app/src/atoms/Slideout/__tests__/Slideout.test.tsx b/app/src/atoms/Slideout/__tests__/Slideout.test.tsx index bd30ce3f578..8e3301ad374 100644 --- a/app/src/atoms/Slideout/__tests__/Slideout.test.tsx +++ b/app/src/atoms/Slideout/__tests__/Slideout.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { screen, fireEvent } from '@testing-library/react' diff --git a/app/src/atoms/SoftwareKeyboard/AlphanumericKeyboard/AlphanumericKeyboard.stories.tsx b/app/src/atoms/SoftwareKeyboard/AlphanumericKeyboard/AlphanumericKeyboard.stories.tsx index dac73384226..5d00c0ac298 100644 --- a/app/src/atoms/SoftwareKeyboard/AlphanumericKeyboard/AlphanumericKeyboard.stories.tsx +++ b/app/src/atoms/SoftwareKeyboard/AlphanumericKeyboard/AlphanumericKeyboard.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useRef } from 'react' import { DIRECTION_COLUMN, Flex, @@ -22,9 +22,9 @@ export default meta type Story = StoryObj const Keyboard = (): JSX.Element => { - const [showKeyboard, setShowKeyboard] = React.useState(false) - const [value, setValue] = React.useState('') - const keyboardRef = React.useRef(null) + const [showKeyboard, setShowKeyboard] = useState(false) + const [value, setValue] = useState('') + const keyboardRef = useRef(null) return (
diff --git a/app/src/atoms/SoftwareKeyboard/FullKeyboard/FullKeyboard.stories.tsx b/app/src/atoms/SoftwareKeyboard/FullKeyboard/FullKeyboard.stories.tsx index 3734a895231..27d79e602f8 100644 --- a/app/src/atoms/SoftwareKeyboard/FullKeyboard/FullKeyboard.stories.tsx +++ b/app/src/atoms/SoftwareKeyboard/FullKeyboard/FullKeyboard.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useRef } from 'react' import { DIRECTION_COLUMN, Flex, @@ -21,9 +21,9 @@ export default meta type Story = StoryObj const Keyboard = (): JSX.Element => { - const [showKeyboard, setShowKeyboard] = React.useState(false) - const [value, setValue] = React.useState('') - const keyboardRef = React.useRef(null) + const [showKeyboard, setShowKeyboard] = useState(false) + const [value, setValue] = useState('') + const keyboardRef = useRef(null) return ( diff --git a/app/src/atoms/SoftwareKeyboard/IndividualKey/IndividualKey.stories.tsx b/app/src/atoms/SoftwareKeyboard/IndividualKey/IndividualKey.stories.tsx index 18978b2318f..e19ee9ecc57 100644 --- a/app/src/atoms/SoftwareKeyboard/IndividualKey/IndividualKey.stories.tsx +++ b/app/src/atoms/SoftwareKeyboard/IndividualKey/IndividualKey.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useRef } from 'react' import { DIRECTION_COLUMN, Flex, @@ -22,9 +22,9 @@ export default meta type Story = StoryObj const Keyboard = ({ ...args }): JSX.Element => { - const [showKeyboard, setShowKeyboard] = React.useState(false) - const [value, setValue] = React.useState('') - const keyboardRef = React.useRef(null) + const [showKeyboard, setShowKeyboard] = useState(false) + const [value, setValue] = useState('') + const keyboardRef = useRef(null) return ( diff --git a/app/src/atoms/SoftwareKeyboard/IndividualKey/index.tsx b/app/src/atoms/SoftwareKeyboard/IndividualKey/index.tsx index 310008cddc8..1ba1ec5e150 100644 --- a/app/src/atoms/SoftwareKeyboard/IndividualKey/index.tsx +++ b/app/src/atoms/SoftwareKeyboard/IndividualKey/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { KeyboardReact as Keyboard } from 'react-simple-keyboard' import type { KeyboardReactInterface } from 'react-simple-keyboard' import '../index.css' diff --git a/app/src/atoms/SoftwareKeyboard/NumericalKeyboard/NumericalKeyboard.stories.tsx b/app/src/atoms/SoftwareKeyboard/NumericalKeyboard/NumericalKeyboard.stories.tsx index 22bca18ca66..216aaac86c5 100644 --- a/app/src/atoms/SoftwareKeyboard/NumericalKeyboard/NumericalKeyboard.stories.tsx +++ b/app/src/atoms/SoftwareKeyboard/NumericalKeyboard/NumericalKeyboard.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useRef } from 'react' import { DIRECTION_COLUMN, Flex, @@ -37,9 +37,9 @@ type Story = StoryObj const Keyboard = (args): JSX.Element => { const { isDecimal, hasHyphen } = args - const [showKeyboard, setShowKeyboard] = React.useState(false) - const [value, setValue] = React.useState('') - const keyboardRef = React.useRef(null) + const [showKeyboard, setShowKeyboard] = useState(false) + const [value, setValue] = useState('') + const keyboardRef = useRef(null) return ( diff --git a/app/src/atoms/SoftwareKeyboard/NumericalKeyboard/index.tsx b/app/src/atoms/SoftwareKeyboard/NumericalKeyboard/index.tsx index 7bb9a73e105..f0025b6c972 100644 --- a/app/src/atoms/SoftwareKeyboard/NumericalKeyboard/index.tsx +++ b/app/src/atoms/SoftwareKeyboard/NumericalKeyboard/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { KeyboardReact as Keyboard } from 'react-simple-keyboard' import { numericalKeyboardLayout, numericalCustom } from '../constants' diff --git a/app/src/atoms/StatusLabel/StatusLabel.stories.tsx b/app/src/atoms/StatusLabel/StatusLabel.stories.tsx index 0be5f54b519..e4ef9485fbf 100644 --- a/app/src/atoms/StatusLabel/StatusLabel.stories.tsx +++ b/app/src/atoms/StatusLabel/StatusLabel.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { COLORS } from '@opentrons/components' import { StatusLabel } from './index' import type { Story, Meta } from '@storybook/react' diff --git a/app/src/atoms/StatusLabel/__tests__/StatusLabel.test.tsx b/app/src/atoms/StatusLabel/__tests__/StatusLabel.test.tsx index 8c7f200d90d..568326f0065 100644 --- a/app/src/atoms/StatusLabel/__tests__/StatusLabel.test.tsx +++ b/app/src/atoms/StatusLabel/__tests__/StatusLabel.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect } from 'vitest' import '@testing-library/jest-dom/vitest' import { screen } from '@testing-library/react' diff --git a/app/src/atoms/StatusLabel/index.tsx b/app/src/atoms/StatusLabel/index.tsx index a1561e5e75d..0cf2af6e384 100644 --- a/app/src/atoms/StatusLabel/index.tsx +++ b/app/src/atoms/StatusLabel/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import capitalize from 'lodash/capitalize' import { diff --git a/app/src/atoms/StepMeter/StepMeter.stories.tsx b/app/src/atoms/StepMeter/StepMeter.stories.tsx index 44fdf79c43f..1491451794b 100644 --- a/app/src/atoms/StepMeter/StepMeter.stories.tsx +++ b/app/src/atoms/StepMeter/StepMeter.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { StepMeter } from './index' diff --git a/app/src/atoms/StepMeter/__tests__/StepMeter.test.tsx b/app/src/atoms/StepMeter/__tests__/StepMeter.test.tsx index 6e8225f738e..90f027b0f7a 100644 --- a/app/src/atoms/StepMeter/__tests__/StepMeter.test.tsx +++ b/app/src/atoms/StepMeter/__tests__/StepMeter.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { screen } from '@testing-library/react' diff --git a/app/src/atoms/StepMeter/index.tsx b/app/src/atoms/StepMeter/index.tsx index 2c2854dc4d6..371e92b9393 100644 --- a/app/src/atoms/StepMeter/index.tsx +++ b/app/src/atoms/StepMeter/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useRef } from 'react' import { css } from 'styled-components' import { Box, @@ -16,7 +16,7 @@ interface StepMeterProps { export const StepMeter = (props: StepMeterProps): JSX.Element => { const { totalSteps, currentStep } = props - const prevPercentComplete = React.useRef(0) + const prevPercentComplete = useRef(0) const progress = currentStep != null ? currentStep : 0 const percentComplete = // this logic puts a cap at 100% percentComplete which we should never run into diff --git a/app/src/atoms/buttons/BackButton.tsx b/app/src/atoms/buttons/BackButton.tsx index 05df2fd800b..29657e1f1b2 100644 --- a/app/src/atoms/buttons/BackButton.tsx +++ b/app/src/atoms/buttons/BackButton.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { useNavigate } from 'react-router-dom' diff --git a/app/src/atoms/buttons/FloatingActionButton.stories.tsx b/app/src/atoms/buttons/FloatingActionButton.stories.tsx index a7526805a20..31e5402c13f 100644 --- a/app/src/atoms/buttons/FloatingActionButton.stories.tsx +++ b/app/src/atoms/buttons/FloatingActionButton.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { ICON_DATA_BY_NAME, VIEWPORT } from '@opentrons/components' import { FloatingActionButton } from './' diff --git a/app/src/atoms/buttons/FloatingActionButton.tsx b/app/src/atoms/buttons/FloatingActionButton.tsx index 76a72cd076b..b7e870eab16 100644 --- a/app/src/atoms/buttons/FloatingActionButton.tsx +++ b/app/src/atoms/buttons/FloatingActionButton.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { diff --git a/app/src/atoms/buttons/IconButton.tsx b/app/src/atoms/buttons/IconButton.tsx index d86324ed843..ee754472ff1 100644 --- a/app/src/atoms/buttons/IconButton.tsx +++ b/app/src/atoms/buttons/IconButton.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { BORDERS, diff --git a/app/src/atoms/buttons/MediumButton.tsx b/app/src/atoms/buttons/MediumButton.tsx index b25ac0432b9..d784029696a 100644 --- a/app/src/atoms/buttons/MediumButton.tsx +++ b/app/src/atoms/buttons/MediumButton.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { ALIGN_CENTER, diff --git a/app/src/atoms/buttons/SmallButton.tsx b/app/src/atoms/buttons/SmallButton.tsx index a177be5a3e0..e659d52fd58 100644 --- a/app/src/atoms/buttons/SmallButton.tsx +++ b/app/src/atoms/buttons/SmallButton.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { ALIGN_CENTER, diff --git a/app/src/atoms/buttons/SubmitPrimaryButton.tsx b/app/src/atoms/buttons/SubmitPrimaryButton.tsx index 246ec6f9051..cdbf3442a65 100644 --- a/app/src/atoms/buttons/SubmitPrimaryButton.tsx +++ b/app/src/atoms/buttons/SubmitPrimaryButton.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { SPACING, diff --git a/app/src/atoms/buttons/TextOnlyButton.tsx b/app/src/atoms/buttons/TextOnlyButton.tsx index 45174218cd9..de3bbc969ab 100644 --- a/app/src/atoms/buttons/TextOnlyButton.tsx +++ b/app/src/atoms/buttons/TextOnlyButton.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Btn, StyledText, COLORS, RESPONSIVENESS } from '@opentrons/components' import type { StyleProps } from '@opentrons/components' import { css } from 'styled-components' diff --git a/app/src/atoms/buttons/ToggleButton.tsx b/app/src/atoms/buttons/ToggleButton.tsx index 5299e332845..b814f45da1d 100644 --- a/app/src/atoms/buttons/ToggleButton.tsx +++ b/app/src/atoms/buttons/ToggleButton.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { Btn, Icon, COLORS, SIZE_1, SIZE_2 } from '@opentrons/components' diff --git a/app/src/atoms/buttons/__tests__/BackButton.test.tsx b/app/src/atoms/buttons/__tests__/BackButton.test.tsx index 6dbe655194a..510abd0ee7d 100644 --- a/app/src/atoms/buttons/__tests__/BackButton.test.tsx +++ b/app/src/atoms/buttons/__tests__/BackButton.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/atoms/buttons/__tests__/FloatingActionButton.test.tsx b/app/src/atoms/buttons/__tests__/FloatingActionButton.test.tsx index 61e147e2542..de2bcd49e24 100644 --- a/app/src/atoms/buttons/__tests__/FloatingActionButton.test.tsx +++ b/app/src/atoms/buttons/__tests__/FloatingActionButton.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { screen } from '@testing-library/react' diff --git a/app/src/atoms/buttons/__tests__/MediumButton.test.tsx b/app/src/atoms/buttons/__tests__/MediumButton.test.tsx index 182c0d80a9c..988248413d9 100644 --- a/app/src/atoms/buttons/__tests__/MediumButton.test.tsx +++ b/app/src/atoms/buttons/__tests__/MediumButton.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import '@testing-library/jest-dom/vitest' import { describe, it, expect, vi, beforeEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/atoms/buttons/__tests__/QuaternaryButton.test.tsx b/app/src/atoms/buttons/__tests__/QuaternaryButton.test.tsx index be8759b1a98..4e10831c73c 100644 --- a/app/src/atoms/buttons/__tests__/QuaternaryButton.test.tsx +++ b/app/src/atoms/buttons/__tests__/QuaternaryButton.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import '@testing-library/jest-dom/vitest' import { describe, it, expect, beforeEach, vi } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/atoms/buttons/__tests__/SmallButton.test.tsx b/app/src/atoms/buttons/__tests__/SmallButton.test.tsx index e39d6f1e850..283f21daf4e 100644 --- a/app/src/atoms/buttons/__tests__/SmallButton.test.tsx +++ b/app/src/atoms/buttons/__tests__/SmallButton.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/atoms/buttons/__tests__/SubmitPrimaryButton.test.tsx b/app/src/atoms/buttons/__tests__/SubmitPrimaryButton.test.tsx index 455e0026cb3..a62ba9c4a95 100644 --- a/app/src/atoms/buttons/__tests__/SubmitPrimaryButton.test.tsx +++ b/app/src/atoms/buttons/__tests__/SubmitPrimaryButton.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/atoms/buttons/__tests__/TertiaryButton.test.tsx b/app/src/atoms/buttons/__tests__/TertiaryButton.test.tsx index 7d653b37881..4a3f95369c8 100644 --- a/app/src/atoms/buttons/__tests__/TertiaryButton.test.tsx +++ b/app/src/atoms/buttons/__tests__/TertiaryButton.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, beforeEach } from 'vitest' import { screen } from '@testing-library/react' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/atoms/buttons/__tests__/ToggleButton.test.tsx b/app/src/atoms/buttons/__tests__/ToggleButton.test.tsx index 23f75531d74..3f61d43c5d0 100644 --- a/app/src/atoms/buttons/__tests__/ToggleButton.test.tsx +++ b/app/src/atoms/buttons/__tests__/ToggleButton.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, vi, beforeEach } from 'vitest' import { screen, fireEvent } from '@testing-library/react' import { COLORS, SIZE_2 } from '@opentrons/components' diff --git a/app/src/atoms/structure/Divider.stories.tsx b/app/src/atoms/structure/Divider.stories.tsx index b38372a575c..747c21bec09 100644 --- a/app/src/atoms/structure/Divider.stories.tsx +++ b/app/src/atoms/structure/Divider.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { ALIGN_CENTER, Box, diff --git a/app/src/atoms/structure/Divider.tsx b/app/src/atoms/structure/Divider.tsx index 2f8eef69cfc..db55ad84f44 100644 --- a/app/src/atoms/structure/Divider.tsx +++ b/app/src/atoms/structure/Divider.tsx @@ -1,13 +1,14 @@ -import * as React from 'react' +import type * as React from 'react' import { Box, COLORS, SPACING } from '@opentrons/components' type Props = React.ComponentProps export function Divider(props: Props): JSX.Element { + const { marginY } = props return ( diff --git a/app/src/atoms/structure/Line.stories.tsx b/app/src/atoms/structure/Line.stories.tsx index 4a37a6ee4f5..0383e5e64a2 100644 --- a/app/src/atoms/structure/Line.stories.tsx +++ b/app/src/atoms/structure/Line.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { ALIGN_CENTER, Box, diff --git a/app/src/atoms/structure/Line.tsx b/app/src/atoms/structure/Line.tsx index 59f0de712bf..ecbbecc24cd 100644 --- a/app/src/atoms/structure/Line.tsx +++ b/app/src/atoms/structure/Line.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Box, BORDERS } from '@opentrons/components' type Props = React.ComponentProps diff --git a/app/src/atoms/structure/__tests__/Divider.test.tsx b/app/src/atoms/structure/__tests__/Divider.test.tsx index 4ee70071ccd..27460be938d 100644 --- a/app/src/atoms/structure/__tests__/Divider.test.tsx +++ b/app/src/atoms/structure/__tests__/Divider.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { screen } from '@testing-library/react' diff --git a/app/src/atoms/structure/__tests__/Line.test.tsx b/app/src/atoms/structure/__tests__/Line.test.tsx index 66f995b768f..d9a9caefba2 100644 --- a/app/src/atoms/structure/__tests__/Line.test.tsx +++ b/app/src/atoms/structure/__tests__/Line.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { screen } from '@testing-library/react' diff --git a/app/src/index.tsx b/app/src/index.tsx index cf4fcbfc44c..9640ffbd7ad 100644 --- a/app/src/index.tsx +++ b/app/src/index.tsx @@ -1,5 +1,4 @@ // client entry point and application manifest -import React from 'react' import ReactDom from 'react-dom/client' import { Provider } from 'react-redux' import { HashRouter } from 'react-router-dom' diff --git a/app/src/organisms/Devices/ProtocolRun/utils/getStandardDeckViewLayerBlockList.ts b/app/src/local-resources/deck_configuration/getStandardDeckViewLayerBlockList.ts similarity index 100% rename from app/src/organisms/Devices/ProtocolRun/utils/getStandardDeckViewLayerBlockList.ts rename to app/src/local-resources/deck_configuration/getStandardDeckViewLayerBlockList.ts diff --git a/app/src/local-resources/deck_configuration/index.ts b/app/src/local-resources/deck_configuration/index.ts new file mode 100644 index 00000000000..9028755b545 --- /dev/null +++ b/app/src/local-resources/deck_configuration/index.ts @@ -0,0 +1 @@ +export * from './getStandardDeckViewLayerBlockList' diff --git a/app/src/resources/instruments/__tests__/hooks.test.ts b/app/src/local-resources/instruments/__tests__/hooks.test.ts similarity index 97% rename from app/src/resources/instruments/__tests__/hooks.test.ts rename to app/src/local-resources/instruments/__tests__/hooks.test.ts index ab21c81525d..94b6043a125 100644 --- a/app/src/resources/instruments/__tests__/hooks.test.ts +++ b/app/src/local-resources/instruments/__tests__/hooks.test.ts @@ -1,6 +1,6 @@ import { beforeEach, describe, expect, it, vi } from 'vitest' -import { useIsOEMMode } from '../../robot-settings/hooks' +import { useIsOEMMode } from '/app/resources/robot-settings/hooks' import { useGripperDisplayName, @@ -11,7 +11,7 @@ import { import type { PipetteV2Specs } from '@opentrons/shared-data' -vi.mock('../../robot-settings/hooks') +vi.mock('/app/resources/robot-settings/hooks') const BRANDED_P1000_FLEX_DISPLAY_NAME = 'Flex 1-Channel 1000 μL' const ANONYMOUS_P1000_FLEX_DISPLAY_NAME = '1-Channel 1000 μL' diff --git a/app/src/resources/instruments/hooks.ts b/app/src/local-resources/instruments/hooks.ts similarity index 96% rename from app/src/resources/instruments/hooks.ts rename to app/src/local-resources/instruments/hooks.ts index 32135d716a4..713dd6f1c83 100644 --- a/app/src/resources/instruments/hooks.ts +++ b/app/src/local-resources/instruments/hooks.ts @@ -5,7 +5,7 @@ import { getPipetteSpecsV2, GRIPPER_MODELS, } from '@opentrons/shared-data' -import { useIsOEMMode } from '../robot-settings/hooks' +import { useIsOEMMode } from '/app/resources/robot-settings/hooks' import type { GripperModel, diff --git a/app/src/local-resources/instruments/index.ts b/app/src/local-resources/instruments/index.ts new file mode 100644 index 00000000000..fc78d35129c --- /dev/null +++ b/app/src/local-resources/instruments/index.ts @@ -0,0 +1 @@ +export * from './hooks' diff --git a/app/src/local-resources/modules/__tests__/getModuleImage.test.ts b/app/src/local-resources/modules/__tests__/getModuleImage.test.ts new file mode 100644 index 00000000000..4fc7b870253 --- /dev/null +++ b/app/src/local-resources/modules/__tests__/getModuleImage.test.ts @@ -0,0 +1,84 @@ +import { describe, it, expect } from 'vitest' +import { getModuleImage } from '../getModuleImage' + +describe('getModuleImage', () => { + it('should render the magnetic module image when the model is a magnetic module gen 1', () => { + const result = getModuleImage('magneticModuleV1') + expect(result).toEqual( + '/app/src/assets/images/magnetic_module_gen_2_transparent.png' + ) + }) + + it('should render the high res magnetic module image when the model is a magnetic module gen 1 high res', () => { + const result = getModuleImage('magneticModuleV1', true) + expect(result).toEqual( + '/app/src/assets/images/modules/magneticModuleV2@3x.png' + ) + }) + + it('should render the magnetic module image when the model is a magnetic module gen 2', () => { + const result = getModuleImage('magneticModuleV2') + expect(result).toEqual( + '/app/src/assets/images/magnetic_module_gen_2_transparent.png' + ) + }) + + it('should render the temperature module image when the model is a temperature module gen 1', () => { + const result = getModuleImage('temperatureModuleV1') + expect(result).toEqual( + '/app/src/assets/images/temp_deck_gen_2_transparent.png' + ) + }) + + it('should render the temperature module image when the model is a temperature module gen 2', () => { + const result = getModuleImage('temperatureModuleV2') + expect(result).toEqual( + '/app/src/assets/images/temp_deck_gen_2_transparent.png' + ) + }) + + it('should render the high res temperature module image when the model is a temperature module high res', () => { + const result = getModuleImage('temperatureModuleV2', true) + expect(result).toEqual( + '/app/src/assets/images/modules/temperatureModuleV2@3x.png' + ) + }) + + it('should render the heater-shaker module image when the model is a heater-shaker module gen 1', () => { + const result = getModuleImage('heaterShakerModuleV1') + expect(result).toEqual( + '/app/src/assets/images/heater_shaker_module_transparent.png' + ) + }) + + it('should render the high res heater-shaker module image when the model is a heater-shaker module gen 1 high res', () => { + const result = getModuleImage('heaterShakerModuleV1', true) + expect(result).toEqual( + '/app/src/assets/images/modules/heaterShakerModuleV1@3x.png' + ) + }) + + it('should render the thermocycler module image when the model is a thermocycler module gen 1', () => { + const result = getModuleImage('thermocyclerModuleV1') + expect(result).toEqual('/app/src/assets/images/thermocycler_closed.png') + }) + + it('should render the high res thermocycler module image when the model is a thermocycler module gen 1 high res', () => { + const result = getModuleImage('thermocyclerModuleV1', true) + expect(result).toEqual( + '/app/src/assets/images/modules/thermocyclerModuleV1@3x.png' + ) + }) + + it('should render the thermocycler module image when the model is a thermocycler module gen 2', () => { + const result = getModuleImage('thermocyclerModuleV2') + expect(result).toEqual( + '/app/src/assets/images/thermocycler_gen_2_closed.png' + ) + }) + + it('should render the magnetic block v1 image when the module is magneticBlockV1', () => { + const result = getModuleImage('magneticBlockV1') + expect(result).toEqual('/app/src/assets/images/magnetic_block_gen_1.png') + }) +}) diff --git a/app/src/local-resources/modules/getModuleImage.ts b/app/src/local-resources/modules/getModuleImage.ts new file mode 100644 index 00000000000..3e8e6a92682 --- /dev/null +++ b/app/src/local-resources/modules/getModuleImage.ts @@ -0,0 +1,36 @@ +import magneticModule from '/app/assets/images/magnetic_module_gen_2_transparent.png' +import temperatureModule from '/app/assets/images/temp_deck_gen_2_transparent.png' +import thermoModuleGen1 from '/app/assets/images/thermocycler_closed.png' +import heaterShakerModule from '/app/assets/images/heater_shaker_module_transparent.png' +import magneticModuleHighRes from '/app/assets/images/modules/magneticModuleV2@3x.png' +import temperatureModuleHighRes from '/app/assets/images/modules/temperatureModuleV2@3x.png' +import thermoModuleGen1HighRes from '/app/assets/images/modules/thermocyclerModuleV1@3x.png' +import heaterShakerModuleHighRes from '/app/assets/images/modules/heaterShakerModuleV1@3x.png' +import thermoModuleGen2 from '/app/assets/images/thermocycler_gen_2_closed.png' +import magneticBlockGen1 from '/app/assets/images/magnetic_block_gen_1.png' + +import type { ModuleModel } from '@opentrons/shared-data' + +export function getModuleImage( + model: ModuleModel, + highRes: boolean = false +): string { + switch (model) { + case 'magneticModuleV1': + case 'magneticModuleV2': + return highRes ? magneticModuleHighRes : magneticModule + case 'temperatureModuleV1': + case 'temperatureModuleV2': + return highRes ? temperatureModuleHighRes : temperatureModule + case 'heaterShakerModuleV1': + return highRes ? heaterShakerModuleHighRes : heaterShakerModule + case 'thermocyclerModuleV1': + return highRes ? thermoModuleGen1HighRes : thermoModuleGen1 + case 'thermocyclerModuleV2': + return thermoModuleGen2 + case 'magneticBlockV1': + return magneticBlockGen1 + default: + return 'Error: unknown module model' + } +} diff --git a/app/src/organisms/Devices/getModulePrepCommands.ts b/app/src/local-resources/modules/getModulePrepCommands.ts similarity index 100% rename from app/src/organisms/Devices/getModulePrepCommands.ts rename to app/src/local-resources/modules/getModulePrepCommands.ts diff --git a/app/src/local-resources/modules/index.ts b/app/src/local-resources/modules/index.ts new file mode 100644 index 00000000000..e508be48e92 --- /dev/null +++ b/app/src/local-resources/modules/index.ts @@ -0,0 +1,2 @@ +export * from './getModulePrepCommands' +export * from './getModuleImage' diff --git a/app/src/molecules/BackgroundOverlay/__tests__/BackgroundOverlay.test.tsx b/app/src/molecules/BackgroundOverlay/__tests__/BackgroundOverlay.test.tsx index 73648cd15b8..e09b3c11765 100644 --- a/app/src/molecules/BackgroundOverlay/__tests__/BackgroundOverlay.test.tsx +++ b/app/src/molecules/BackgroundOverlay/__tests__/BackgroundOverlay.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, vi } from 'vitest' import { fireEvent, screen } from '@testing-library/react' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/molecules/BackgroundOverlay/index.tsx b/app/src/molecules/BackgroundOverlay/index.tsx index ccfdb273fc4..3d6c6d976c6 100644 --- a/app/src/molecules/BackgroundOverlay/index.tsx +++ b/app/src/molecules/BackgroundOverlay/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { COLORS, Flex, POSITION_FIXED } from '@opentrons/components' diff --git a/app/src/molecules/CardButton/CardButton.stories.tsx b/app/src/molecules/CardButton/CardButton.stories.tsx index 67256590b4b..3d80e18a3bb 100644 --- a/app/src/molecules/CardButton/CardButton.stories.tsx +++ b/app/src/molecules/CardButton/CardButton.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Flex, ICON_DATA_BY_NAME, diff --git a/app/src/molecules/CardButton/__tests__/CardButton.test.tsx b/app/src/molecules/CardButton/__tests__/CardButton.test.tsx index ceff9582c21..820dfbdc4e9 100644 --- a/app/src/molecules/CardButton/__tests__/CardButton.test.tsx +++ b/app/src/molecules/CardButton/__tests__/CardButton.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import '@testing-library/jest-dom/vitest' import { describe, it, expect, vi, beforeEach } from 'vitest' diff --git a/app/src/molecules/CardButton/index.tsx b/app/src/molecules/CardButton/index.tsx index 2b8a6293198..86ed6e3731a 100644 --- a/app/src/molecules/CardButton/index.tsx +++ b/app/src/molecules/CardButton/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useNavigate } from 'react-router-dom' import { css } from 'styled-components' import { diff --git a/app/src/molecules/CollapsibleSection/__tests__/CollapsibleSection.test.tsx b/app/src/molecules/CollapsibleSection/__tests__/CollapsibleSection.test.tsx index 2a444b25a65..3acef8b4a29 100644 --- a/app/src/molecules/CollapsibleSection/__tests__/CollapsibleSection.test.tsx +++ b/app/src/molecules/CollapsibleSection/__tests__/CollapsibleSection.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import '@testing-library/jest-dom/vitest' import { describe, it, expect } from 'vitest' import { fireEvent, render, screen } from '@testing-library/react' diff --git a/app/src/molecules/Command/Command.stories.tsx b/app/src/molecules/Command/Command.stories.tsx index 6cba996f21a..1fe64215ef3 100644 --- a/app/src/molecules/Command/Command.stories.tsx +++ b/app/src/molecules/Command/Command.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Command as CommandComponent } from '.' import type { CommandState } from './Command' import * as Fixtures from './__fixtures__' diff --git a/app/src/molecules/Command/Command.tsx b/app/src/molecules/Command/Command.tsx index cf7e5ce0d92..32d95a33371 100644 --- a/app/src/molecules/Command/Command.tsx +++ b/app/src/molecules/Command/Command.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Flex, JUSTIFY_CENTER, diff --git a/app/src/molecules/Command/CommandIcon.stories.tsx b/app/src/molecules/Command/CommandIcon.stories.tsx index 4312d585554..4452a67cefb 100644 --- a/app/src/molecules/Command/CommandIcon.stories.tsx +++ b/app/src/molecules/Command/CommandIcon.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { CommandIcon as CommandIconComponent } from './CommandIcon' import type { ICON_BY_COMMAND_TYPE } from './CommandIcon' diff --git a/app/src/molecules/Command/CommandIcon.tsx b/app/src/molecules/Command/CommandIcon.tsx index 77eb4ef3914..4002de50973 100644 --- a/app/src/molecules/Command/CommandIcon.tsx +++ b/app/src/molecules/Command/CommandIcon.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Icon } from '@opentrons/components' import type { IconName, StyleProps } from '@opentrons/components' import type { RunTimeCommand } from '@opentrons/shared-data' diff --git a/app/src/molecules/Command/CommandIndex.tsx b/app/src/molecules/Command/CommandIndex.tsx index bcabae9ef90..6357d372486 100644 --- a/app/src/molecules/Command/CommandIndex.tsx +++ b/app/src/molecules/Command/CommandIndex.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - import { StyledText, RESPONSIVENESS, diff --git a/app/src/molecules/Command/CommandText.stories.tsx b/app/src/molecules/Command/CommandText.stories.tsx index fa361be59dc..f6b40e0a4e4 100644 --- a/app/src/molecules/Command/CommandText.stories.tsx +++ b/app/src/molecules/Command/CommandText.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Box } from '@opentrons/components' import { CommandText as CommandTextComponent } from '.' import type { RobotType } from '@opentrons/shared-data' diff --git a/app/src/molecules/Command/CommandText.tsx b/app/src/molecules/Command/CommandText.tsx index 75b12733eca..70fb5281817 100644 --- a/app/src/molecules/Command/CommandText.tsx +++ b/app/src/molecules/Command/CommandText.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { pick } from 'lodash' import { diff --git a/app/src/molecules/Command/__tests__/CommandText.test.tsx b/app/src/molecules/Command/__tests__/CommandText.test.tsx index 484c9ee0323..7425c2e1853 100644 --- a/app/src/molecules/Command/__tests__/CommandText.test.tsx +++ b/app/src/molecules/Command/__tests__/CommandText.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { it, expect, describe } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/molecules/Command/hooks/useCommandTextString/utils/getConfigureNozzleLayoutCommandText.ts b/app/src/molecules/Command/hooks/useCommandTextString/utils/getConfigureNozzleLayoutCommandText.ts index e6693a4b937..04d476fadd1 100644 --- a/app/src/molecules/Command/hooks/useCommandTextString/utils/getConfigureNozzleLayoutCommandText.ts +++ b/app/src/molecules/Command/hooks/useCommandTextString/utils/getConfigureNozzleLayoutCommandText.ts @@ -13,8 +13,17 @@ export function getConfigureNozzleLayoutCommandText({ pip => pip.id === pipetteId )?.pipetteName + // TODO(cb, 2024-09-10): confirm these strings for copy consistency and add them to i18n + const ConfigAmount = { + SINGLE: 'single nozzle layout', + COLUMN: 'column layout', + ROW: 'row layout', + QUADRANT: 'partial layout', + ALL: 'all nozzles', + } + return t('configure_nozzle_layout', { - amount: configurationParams.style === 'COLUMN' ? '8' : 'all', + layout: ConfigAmount[configurationParams.style], pipette: pipetteName != null ? getPipetteSpecsV2(pipetteName)?.displayName : '', }) diff --git a/app/src/molecules/FileUpload/FileUpload.stories.tsx b/app/src/molecules/FileUpload/FileUpload.stories.tsx index 5ddaab2698f..855a50c5629 100644 --- a/app/src/molecules/FileUpload/FileUpload.stories.tsx +++ b/app/src/molecules/FileUpload/FileUpload.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import testFile from './__tests__/test-file.png' import { FileUpload } from '.' diff --git a/app/src/molecules/FileUpload/__tests__/FileUpload.test.tsx b/app/src/molecules/FileUpload/__tests__/FileUpload.test.tsx index 4e18cdfb08a..cd5b5adfd82 100644 --- a/app/src/molecules/FileUpload/__tests__/FileUpload.test.tsx +++ b/app/src/molecules/FileUpload/__tests__/FileUpload.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { beforeEach, describe, expect, it, vi } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/molecules/FileUpload/index.tsx b/app/src/molecules/FileUpload/index.tsx index dc953027235..777d186ccb0 100644 --- a/app/src/molecules/FileUpload/index.tsx +++ b/app/src/molecules/FileUpload/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { diff --git a/app/src/molecules/GenericWizardTile/GenericWizardTile.stories.tsx b/app/src/molecules/GenericWizardTile/GenericWizardTile.stories.tsx index b707286cb55..5444feff20b 100644 --- a/app/src/molecules/GenericWizardTile/GenericWizardTile.stories.tsx +++ b/app/src/molecules/GenericWizardTile/GenericWizardTile.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Provider } from 'react-redux' import { createStore } from 'redux' import { diff --git a/app/src/molecules/GenericWizardTile/__tests__/GenericWizardTile.test.tsx b/app/src/molecules/GenericWizardTile/__tests__/GenericWizardTile.test.tsx index 8070a4d3cd7..1f53800ff6d 100644 --- a/app/src/molecules/GenericWizardTile/__tests__/GenericWizardTile.test.tsx +++ b/app/src/molecules/GenericWizardTile/__tests__/GenericWizardTile.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import '@testing-library/jest-dom/vitest' import { describe, it, expect, vi, beforeEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/molecules/GenericWizardTile/index.tsx b/app/src/molecules/GenericWizardTile/index.tsx index 1d07d2bb9ea..4bda416cca5 100644 --- a/app/src/molecules/GenericWizardTile/index.tsx +++ b/app/src/molecules/GenericWizardTile/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useSelector } from 'react-redux' import styled, { css } from 'styled-components' import { useTranslation } from 'react-i18next' diff --git a/app/src/molecules/InProgressModal/InProgressModal.stories.tsx b/app/src/molecules/InProgressModal/InProgressModal.stories.tsx index 9fc5f5b30c9..cfa241a95fc 100644 --- a/app/src/molecules/InProgressModal/InProgressModal.stories.tsx +++ b/app/src/molecules/InProgressModal/InProgressModal.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import type { Meta, StoryObj } from '@storybook/react' import { InProgressModal as InProgressModalComponent } from './' import { SimpleWizardInProgressBody } from '../SimpleWizardBody' diff --git a/app/src/molecules/InProgressModal/InProgressModal.tsx b/app/src/molecules/InProgressModal/InProgressModal.tsx index 24c5d9a8f71..c6fefe761a2 100644 --- a/app/src/molecules/InProgressModal/InProgressModal.tsx +++ b/app/src/molecules/InProgressModal/InProgressModal.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { ALIGN_CENTER, diff --git a/app/src/molecules/InProgressModal/__tests__/InProgressModal.test.tsx b/app/src/molecules/InProgressModal/__tests__/InProgressModal.test.tsx index 9d0f0874874..f670fa221c3 100644 --- a/app/src/molecules/InProgressModal/__tests__/InProgressModal.test.tsx +++ b/app/src/molecules/InProgressModal/__tests__/InProgressModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, beforeEach, vi } from 'vitest' import { i18n } from '/app/i18n' diff --git a/app/src/molecules/InfoMessage/__tests__/InfoMessage.test.tsx b/app/src/molecules/InfoMessage/__tests__/InfoMessage.test.tsx index c4327de590b..5ff6948976f 100644 --- a/app/src/molecules/InfoMessage/__tests__/InfoMessage.test.tsx +++ b/app/src/molecules/InfoMessage/__tests__/InfoMessage.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, beforeEach } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' import { screen } from '@testing-library/react' diff --git a/app/src/molecules/InfoMessage/index.tsx b/app/src/molecules/InfoMessage/index.tsx index 618233b0a20..c5d8bda1666 100644 --- a/app/src/molecules/InfoMessage/index.tsx +++ b/app/src/molecules/InfoMessage/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { ALIGN_CENTER, ALIGN_FLEX_START, diff --git a/app/src/molecules/InstrumentCard/InstrumentCard.stories.tsx b/app/src/molecules/InstrumentCard/InstrumentCard.stories.tsx index 7a56a1eba9c..7db21c46afc 100644 --- a/app/src/molecules/InstrumentCard/InstrumentCard.stories.tsx +++ b/app/src/molecules/InstrumentCard/InstrumentCard.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { InstrumentCard } from './' import type { Story, Meta } from '@storybook/react' diff --git a/app/src/molecules/InstrumentCard/__tests__/InstrumentCard.test.tsx b/app/src/molecules/InstrumentCard/__tests__/InstrumentCard.test.tsx index 6efa70a7752..ca5905829f0 100644 --- a/app/src/molecules/InstrumentCard/__tests__/InstrumentCard.test.tsx +++ b/app/src/molecules/InstrumentCard/__tests__/InstrumentCard.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, render, screen } from '@testing-library/react' import '@testing-library/jest-dom/vitest' import { describe, it, expect, vi } from 'vitest' diff --git a/app/src/molecules/InstrumentCard/index.tsx b/app/src/molecules/InstrumentCard/index.tsx index 4384db759e3..bcb2ccb47ac 100644 --- a/app/src/molecules/InstrumentCard/index.tsx +++ b/app/src/molecules/InstrumentCard/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { ALIGN_CENTER, diff --git a/app/src/molecules/InterventionModal/CategorizedStepContent.stories.tsx b/app/src/molecules/InterventionModal/CategorizedStepContent.stories.tsx index 4037532cf6f..04c04c0f3c6 100644 --- a/app/src/molecules/InterventionModal/CategorizedStepContent.stories.tsx +++ b/app/src/molecules/InterventionModal/CategorizedStepContent.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { CategorizedStepContent, TwoColumn } from '.' diff --git a/app/src/molecules/InterventionModal/CategorizedStepContent.tsx b/app/src/molecules/InterventionModal/CategorizedStepContent.tsx index a9ec14b6876..f0b78a256af 100644 --- a/app/src/molecules/InterventionModal/CategorizedStepContent.tsx +++ b/app/src/molecules/InterventionModal/CategorizedStepContent.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { Command, CommandIndex } from '../Command' import type { NonSkeletonCommandState, CommandTextData } from '../Command' diff --git a/app/src/molecules/InterventionModal/DeckMapContent.stories.tsx b/app/src/molecules/InterventionModal/DeckMapContent.stories.tsx index 56c896655e0..a41339004e4 100644 --- a/app/src/molecules/InterventionModal/DeckMapContent.stories.tsx +++ b/app/src/molecules/InterventionModal/DeckMapContent.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { DeckMapContent } from '.' diff --git a/app/src/molecules/InterventionModal/DescriptionContent.tsx b/app/src/molecules/InterventionModal/DescriptionContent.tsx index 8f166ad79c6..a3da661da0e 100644 --- a/app/src/molecules/InterventionModal/DescriptionContent.tsx +++ b/app/src/molecules/InterventionModal/DescriptionContent.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Flex, SPACING, diff --git a/app/src/molecules/InterventionModal/InterventionContent/InterventionContent.stories.tsx b/app/src/molecules/InterventionModal/InterventionContent/InterventionContent.stories.tsx index d910847c2d3..47cfea68317 100644 --- a/app/src/molecules/InterventionModal/InterventionContent/InterventionContent.stories.tsx +++ b/app/src/molecules/InterventionModal/InterventionContent/InterventionContent.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { ICON_DATA_BY_NAME } from '@opentrons/components' import { InterventionContent } from '.' import { TwoColumn } from '../TwoColumn' diff --git a/app/src/molecules/InterventionModal/InterventionContent/InterventionInfo.stories.tsx b/app/src/molecules/InterventionModal/InterventionContent/InterventionInfo.stories.tsx index caac9a06d5c..33a6ddbcac1 100644 --- a/app/src/molecules/InterventionModal/InterventionContent/InterventionInfo.stories.tsx +++ b/app/src/molecules/InterventionModal/InterventionContent/InterventionInfo.stories.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - import { Box, ICON_DATA_BY_NAME } from '@opentrons/components' import { InterventionInfo } from './InterventionInfo' diff --git a/app/src/molecules/InterventionModal/InterventionContent/InterventionInfo.tsx b/app/src/molecules/InterventionModal/InterventionContent/InterventionInfo.tsx index f24efb492be..03435fb48b5 100644 --- a/app/src/molecules/InterventionModal/InterventionContent/InterventionInfo.tsx +++ b/app/src/molecules/InterventionModal/InterventionContent/InterventionInfo.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { diff --git a/app/src/molecules/InterventionModal/InterventionContent/index.tsx b/app/src/molecules/InterventionModal/InterventionContent/index.tsx index 63e1e33e2c6..8d2bbba9c8c 100644 --- a/app/src/molecules/InterventionModal/InterventionContent/index.tsx +++ b/app/src/molecules/InterventionModal/InterventionContent/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Flex, StyledText, diff --git a/app/src/molecules/InterventionModal/InterventionModal.stories.tsx b/app/src/molecules/InterventionModal/InterventionModal.stories.tsx index 63f2827cc1e..75b025faaf7 100644 --- a/app/src/molecules/InterventionModal/InterventionModal.stories.tsx +++ b/app/src/molecules/InterventionModal/InterventionModal.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Provider } from 'react-redux' import { createStore } from 'redux' diff --git a/app/src/molecules/InterventionModal/ModalContentMixed.tsx b/app/src/molecules/InterventionModal/ModalContentMixed.tsx index 08c5787b1db..4c41003c3d8 100644 --- a/app/src/molecules/InterventionModal/ModalContentMixed.tsx +++ b/app/src/molecules/InterventionModal/ModalContentMixed.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Icon, Flex, diff --git a/app/src/molecules/InterventionModal/ModalContentOneColSimpleButtons.stories.tsx b/app/src/molecules/InterventionModal/ModalContentOneColSimpleButtons.stories.tsx index 731024ef5e2..626b6bf5243 100644 --- a/app/src/molecules/InterventionModal/ModalContentOneColSimpleButtons.stories.tsx +++ b/app/src/molecules/InterventionModal/ModalContentOneColSimpleButtons.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { ModalContentOneColSimpleButtons as ModalContentOneColSimpleButtonsComponent } from './ModalContentOneColSimpleButtons' import type { Meta, StoryObj } from '@storybook/react' diff --git a/app/src/molecules/InterventionModal/OneColumn.stories.tsx b/app/src/molecules/InterventionModal/OneColumn.stories.tsx index ef4e8a6a02f..156eb44b514 100644 --- a/app/src/molecules/InterventionModal/OneColumn.stories.tsx +++ b/app/src/molecules/InterventionModal/OneColumn.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Box, RESPONSIVENESS } from '@opentrons/components' diff --git a/app/src/molecules/InterventionModal/OneColumn.tsx b/app/src/molecules/InterventionModal/OneColumn.tsx index e92f3ffd51e..35a37dd10f9 100644 --- a/app/src/molecules/InterventionModal/OneColumn.tsx +++ b/app/src/molecules/InterventionModal/OneColumn.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Flex, diff --git a/app/src/molecules/InterventionModal/OneColumnOrTwoColumn.stories.tsx b/app/src/molecules/InterventionModal/OneColumnOrTwoColumn.stories.tsx index 791edcbdb83..95e9ac3ff98 100644 --- a/app/src/molecules/InterventionModal/OneColumnOrTwoColumn.stories.tsx +++ b/app/src/molecules/InterventionModal/OneColumnOrTwoColumn.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { OneColumnOrTwoColumn } from './' diff --git a/app/src/molecules/InterventionModal/OneColumnOrTwoColumn.tsx b/app/src/molecules/InterventionModal/OneColumnOrTwoColumn.tsx index 8a6455d67e3..db25d343be5 100644 --- a/app/src/molecules/InterventionModal/OneColumnOrTwoColumn.tsx +++ b/app/src/molecules/InterventionModal/OneColumnOrTwoColumn.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { diff --git a/app/src/molecules/InterventionModal/TwoColumn.stories.tsx b/app/src/molecules/InterventionModal/TwoColumn.stories.tsx index a43bfb13068..a1d83ca750d 100644 --- a/app/src/molecules/InterventionModal/TwoColumn.stories.tsx +++ b/app/src/molecules/InterventionModal/TwoColumn.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import SuccessIcon from '/app/assets/images/icon_success.png' diff --git a/app/src/molecules/InterventionModal/TwoColumn.tsx b/app/src/molecules/InterventionModal/TwoColumn.tsx index f0ed10ebf2a..fc9072232be 100644 --- a/app/src/molecules/InterventionModal/TwoColumn.tsx +++ b/app/src/molecules/InterventionModal/TwoColumn.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Flex, Box, DIRECTION_ROW, SPACING, WRAP } from '@opentrons/components' import type { StyleProps } from '@opentrons/components' diff --git a/app/src/molecules/InterventionModal/__tests__/InterventionModal.test.tsx b/app/src/molecules/InterventionModal/__tests__/InterventionModal.test.tsx index 58297e6ec80..a063bee13bc 100644 --- a/app/src/molecules/InterventionModal/__tests__/InterventionModal.test.tsx +++ b/app/src/molecules/InterventionModal/__tests__/InterventionModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, describe, it, expect, beforeEach } from 'vitest' import { when } from 'vitest-when' import '@testing-library/jest-dom/vitest' diff --git a/app/src/molecules/InterventionModal/__tests__/ModalContentOneColSimpleButtons.test.tsx b/app/src/molecules/InterventionModal/__tests__/ModalContentOneColSimpleButtons.test.tsx index 236b236b390..27a40d82ad4 100644 --- a/app/src/molecules/InterventionModal/__tests__/ModalContentOneColSimpleButtons.test.tsx +++ b/app/src/molecules/InterventionModal/__tests__/ModalContentOneColSimpleButtons.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, describe, it, expect } from 'vitest' import { render, screen, fireEvent } from '@testing-library/react' diff --git a/app/src/molecules/InterventionModal/index.tsx b/app/src/molecules/InterventionModal/index.tsx index 3d27946b2f7..a4e0c19c618 100644 --- a/app/src/molecules/InterventionModal/index.tsx +++ b/app/src/molecules/InterventionModal/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useSelector } from 'react-redux' import { css } from 'styled-components' diff --git a/app/src/molecules/InterventionModal/story-utils/StandIn.tsx b/app/src/molecules/InterventionModal/story-utils/StandIn.tsx index 0fb46f44b8c..33eb868816d 100644 --- a/app/src/molecules/InterventionModal/story-utils/StandIn.tsx +++ b/app/src/molecules/InterventionModal/story-utils/StandIn.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Box, BORDERS } from '@opentrons/components' export function StandInContent({ diff --git a/app/src/molecules/InterventionModal/story-utils/VisibleContainer.tsx b/app/src/molecules/InterventionModal/story-utils/VisibleContainer.tsx index b716b3335ee..8701f074c8a 100644 --- a/app/src/molecules/InterventionModal/story-utils/VisibleContainer.tsx +++ b/app/src/molecules/InterventionModal/story-utils/VisibleContainer.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - import { Box, BORDERS, SPACING } from '@opentrons/components' import type { StyleProps } from '@opentrons/components' diff --git a/app/src/molecules/JogControls/ControlContainer.tsx b/app/src/molecules/JogControls/ControlContainer.tsx index 6d47b3df405..02e3c0b1692 100644 --- a/app/src/molecules/JogControls/ControlContainer.tsx +++ b/app/src/molecules/JogControls/ControlContainer.tsx @@ -1,4 +1,4 @@ -import React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { BORDERS, diff --git a/app/src/molecules/JogControls/JogControls.stories.tsx b/app/src/molecules/JogControls/JogControls.stories.tsx index a4008ea6256..55b84d51947 100644 --- a/app/src/molecules/JogControls/JogControls.stories.tsx +++ b/app/src/molecules/JogControls/JogControls.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { JogControls } from './index' diff --git a/app/src/molecules/JogControls/StepSizeControl.tsx b/app/src/molecules/JogControls/StepSizeControl.tsx index 69e290677b3..77d1e65b870 100644 --- a/app/src/molecules/JogControls/StepSizeControl.tsx +++ b/app/src/molecules/JogControls/StepSizeControl.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import styled, { css } from 'styled-components' import { diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareStackModal.tsx b/app/src/molecules/LabwareStackModal/LabwareStackModal.tsx similarity index 97% rename from app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareStackModal.tsx rename to app/src/molecules/LabwareStackModal/LabwareStackModal.tsx index a50be194f45..9c6602023a8 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareStackModal.tsx +++ b/app/src/molecules/LabwareStackModal/LabwareStackModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' import { css } from 'styled-components' @@ -18,10 +17,12 @@ import { } from '@opentrons/components' import { OddModal } from '/app/molecules/OddModal' import { getIsOnDevice } from '/app/redux/config' -import { getLocationInfoNames } from '../utils/getLocationInfoNames' -import { getSlotLabwareDefinition } from '../utils/getSlotLabwareDefinition' +import { + getLocationInfoNames, + getSlotLabwareDefinition, +} from '/app/transformations/commands' import { Divider } from '/app/atoms/structure' -import { getModuleImage } from '../SetupModuleAndDeck/utils' +import { getModuleImage } from '/app/local-resources/modules' import { FLEX_ROBOT_TYPE, getModuleDisplayName, diff --git a/app/src/molecules/LabwareStackModal/index.ts b/app/src/molecules/LabwareStackModal/index.ts new file mode 100644 index 00000000000..23f4662e721 --- /dev/null +++ b/app/src/molecules/LabwareStackModal/index.ts @@ -0,0 +1 @@ +export * from './LabwareStackModal' diff --git a/app/src/molecules/MiniCard/MiniCard.stories.tsx b/app/src/molecules/MiniCard/MiniCard.stories.tsx index ba24854ab4d..8045c567d13 100644 --- a/app/src/molecules/MiniCard/MiniCard.stories.tsx +++ b/app/src/molecules/MiniCard/MiniCard.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { ALIGN_CENTER, Box, diff --git a/app/src/molecules/MiniCard/__tests__/MiniCard.test.tsx b/app/src/molecules/MiniCard/__tests__/MiniCard.test.tsx index 93436e0d37e..fcc76f38505 100644 --- a/app/src/molecules/MiniCard/__tests__/MiniCard.test.tsx +++ b/app/src/molecules/MiniCard/__tests__/MiniCard.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/molecules/MiniCard/index.tsx b/app/src/molecules/MiniCard/index.tsx index 526b2076ef4..b65ccbbb59d 100644 --- a/app/src/molecules/MiniCard/index.tsx +++ b/app/src/molecules/MiniCard/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { BORDERS, diff --git a/app/src/molecules/ModuleIcon/ModuleIcon.stories.tsx b/app/src/molecules/ModuleIcon/ModuleIcon.stories.tsx index 06484370bac..002c0971e8f 100644 --- a/app/src/molecules/ModuleIcon/ModuleIcon.stories.tsx +++ b/app/src/molecules/ModuleIcon/ModuleIcon.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { DIRECTION_COLUMN, diff --git a/app/src/molecules/ModuleIcon/__tests__/ModuleIcon.test.tsx b/app/src/molecules/ModuleIcon/__tests__/ModuleIcon.test.tsx index 081d08cd4c3..73c44639e51 100644 --- a/app/src/molecules/ModuleIcon/__tests__/ModuleIcon.test.tsx +++ b/app/src/molecules/ModuleIcon/__tests__/ModuleIcon.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { COLORS, SPACING } from '@opentrons/components' import { describe, it, expect, vi, beforeEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/molecules/ModuleIcon/index.tsx b/app/src/molecules/ModuleIcon/index.tsx index ed1ae3bcaa4..671b7b9210c 100644 --- a/app/src/molecules/ModuleIcon/index.tsx +++ b/app/src/molecules/ModuleIcon/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { diff --git a/app/src/organisms/Devices/ModuleInfo.tsx b/app/src/molecules/ModuleInfo/ModuleInfo.tsx similarity index 97% rename from app/src/organisms/Devices/ModuleInfo.tsx rename to app/src/molecules/ModuleInfo/ModuleInfo.tsx index 30cd3f86f7a..3e68079bd5f 100644 --- a/app/src/organisms/Devices/ModuleInfo.tsx +++ b/app/src/molecules/ModuleInfo/ModuleInfo.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { ALIGN_CENTER, @@ -19,7 +18,7 @@ import { MAGNETIC_BLOCK_V1, } from '@opentrons/shared-data' -import { useRunHasStarted } from './hooks' +import { useRunHasStarted } from '/app/resources/runs' import type { ModuleModel } from '@opentrons/shared-data' import type { PhysicalPort } from '/app/redux/modules/api-types' diff --git a/app/src/organisms/Devices/__tests__/ModuleInfo.test.tsx b/app/src/molecules/ModuleInfo/__tests__/ModuleInfo.test.tsx similarity index 95% rename from app/src/organisms/Devices/__tests__/ModuleInfo.test.tsx rename to app/src/molecules/ModuleInfo/__tests__/ModuleInfo.test.tsx index 68e2ff8660b..1d732faeb18 100644 --- a/app/src/organisms/Devices/__tests__/ModuleInfo.test.tsx +++ b/app/src/molecules/ModuleInfo/__tests__/ModuleInfo.test.tsx @@ -1,4 +1,4 @@ -import React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' import '@testing-library/jest-dom/vitest' @@ -6,10 +6,10 @@ import { renderWithProviders } from '/app/__testing-utils__' import { when } from 'vitest-when' import { i18n } from '/app/i18n' import { ModuleInfo } from '../ModuleInfo' -import { useRunHasStarted } from '../hooks' +import { useRunHasStarted } from '/app/resources/runs' import type { ModuleModel, ModuleType } from '@opentrons/shared-data' -vi.mock('../hooks') +vi.mock('/app/resources/runs') const render = (props: React.ComponentProps) => { return renderWithProviders(, { diff --git a/app/src/molecules/ModuleInfo/index.ts b/app/src/molecules/ModuleInfo/index.ts new file mode 100644 index 00000000000..0fdce59dc8a --- /dev/null +++ b/app/src/molecules/ModuleInfo/index.ts @@ -0,0 +1 @@ +export * from './ModuleInfo' diff --git a/app/src/molecules/NavTab/NavTab.stories.tsx b/app/src/molecules/NavTab/NavTab.stories.tsx index a9b54818b51..d4847a1ed0f 100644 --- a/app/src/molecules/NavTab/NavTab.stories.tsx +++ b/app/src/molecules/NavTab/NavTab.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { Flex, diff --git a/app/src/molecules/NavTab/__tests__/NavTab.test.tsx b/app/src/molecules/NavTab/__tests__/NavTab.test.tsx index 6345c2fac7d..176c76b60cc 100644 --- a/app/src/molecules/NavTab/__tests__/NavTab.test.tsx +++ b/app/src/molecules/NavTab/__tests__/NavTab.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import '@testing-library/jest-dom/vitest' import { describe, it, expect, beforeEach } from 'vitest' diff --git a/app/src/molecules/NavTab/index.tsx b/app/src/molecules/NavTab/index.tsx index 97d6e4a9f12..e170da56e43 100644 --- a/app/src/molecules/NavTab/index.tsx +++ b/app/src/molecules/NavTab/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import styled, { css } from 'styled-components' import { NavLink } from 'react-router-dom' diff --git a/app/src/molecules/ODDBackButton/ODDBackButton.stories.tsx b/app/src/molecules/ODDBackButton/ODDBackButton.stories.tsx index 6fad4d7ae4a..32719ce1665 100644 --- a/app/src/molecules/ODDBackButton/ODDBackButton.stories.tsx +++ b/app/src/molecules/ODDBackButton/ODDBackButton.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { VIEWPORT } from '@opentrons/components' import { ODDBackButton } from '.' import type { Story, Meta } from '@storybook/react' diff --git a/app/src/molecules/ODDBackButton/__tests__/ODDBackButton.test.tsx b/app/src/molecules/ODDBackButton/__tests__/ODDBackButton.test.tsx index 5e186acf9cd..2b520279cf8 100644 --- a/app/src/molecules/ODDBackButton/__tests__/ODDBackButton.test.tsx +++ b/app/src/molecules/ODDBackButton/__tests__/ODDBackButton.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import '@testing-library/jest-dom/vitest' import { describe, it, expect, vi, beforeEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/molecules/ODDBackButton/index.tsx b/app/src/molecules/ODDBackButton/index.tsx index 9dff1968160..396feaa5e19 100644 --- a/app/src/molecules/ODDBackButton/index.tsx +++ b/app/src/molecules/ODDBackButton/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { ALIGN_CENTER, diff --git a/app/src/molecules/OddModal/ModalHeader.stories.tsx b/app/src/molecules/OddModal/ModalHeader.stories.tsx index adf9524988e..d6e5d95c598 100644 --- a/app/src/molecules/OddModal/ModalHeader.stories.tsx +++ b/app/src/molecules/OddModal/ModalHeader.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { COLORS, VIEWPORT } from '@opentrons/components' import { OddModalHeader } from './OddModalHeader' import type { Story, Meta } from '@storybook/react' diff --git a/app/src/molecules/OddModal/OddModal.stories.tsx b/app/src/molecules/OddModal/OddModal.stories.tsx index 1ba0914779b..1210ad31005 100644 --- a/app/src/molecules/OddModal/OddModal.stories.tsx +++ b/app/src/molecules/OddModal/OddModal.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { COLORS, Flex, BORDERS, SPACING, VIEWPORT } from '@opentrons/components' import { OddModal } from './OddModal' import type { Story, Meta } from '@storybook/react' diff --git a/app/src/molecules/OddModal/OddModal.tsx b/app/src/molecules/OddModal/OddModal.tsx index 1f18e3e1fd9..9b32974000c 100644 --- a/app/src/molecules/OddModal/OddModal.tsx +++ b/app/src/molecules/OddModal/OddModal.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { ALIGN_CENTER, BORDERS, diff --git a/app/src/molecules/OddModal/OddModalHeader.tsx b/app/src/molecules/OddModal/OddModalHeader.tsx index cd79d1a81f5..5544a14ecbc 100644 --- a/app/src/molecules/OddModal/OddModalHeader.tsx +++ b/app/src/molecules/OddModal/OddModalHeader.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { ALIGN_CENTER, BORDERS, diff --git a/app/src/molecules/OddModal/SmallModalChildren.stories.tsx b/app/src/molecules/OddModal/SmallModalChildren.stories.tsx index c1889ca718e..a4a0bd9bcc5 100644 --- a/app/src/molecules/OddModal/SmallModalChildren.stories.tsx +++ b/app/src/molecules/OddModal/SmallModalChildren.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { VIEWPORT } from '@opentrons/components' import { SmallModalChildren } from './SmallModalChildren' import type { Story, Meta } from '@storybook/react' diff --git a/app/src/molecules/OddModal/SmallModalChildren.tsx b/app/src/molecules/OddModal/SmallModalChildren.tsx index 6ceb1cf50ca..91345e1fea7 100644 --- a/app/src/molecules/OddModal/SmallModalChildren.tsx +++ b/app/src/molecules/OddModal/SmallModalChildren.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { COLORS, DIRECTION_COLUMN, diff --git a/app/src/molecules/OddModal/__tests__/OddModal.test.tsx b/app/src/molecules/OddModal/__tests__/OddModal.test.tsx index 4c289650fd8..d8c129a8b01 100644 --- a/app/src/molecules/OddModal/__tests__/OddModal.test.tsx +++ b/app/src/molecules/OddModal/__tests__/OddModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import '@testing-library/jest-dom/vitest' import { describe, it, expect, vi, beforeEach } from 'vitest' diff --git a/app/src/molecules/OddModal/__tests__/OddModalHeader.test.tsx b/app/src/molecules/OddModal/__tests__/OddModalHeader.test.tsx index 3c3cb72f8b2..e824e49648c 100644 --- a/app/src/molecules/OddModal/__tests__/OddModalHeader.test.tsx +++ b/app/src/molecules/OddModal/__tests__/OddModalHeader.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import '@testing-library/jest-dom/vitest' import { describe, it, expect, vi, beforeEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/molecules/OddModal/__tests__/SmallModalChildren.test.tsx b/app/src/molecules/OddModal/__tests__/SmallModalChildren.test.tsx index 0025a806fe2..77a6e56095b 100644 --- a/app/src/molecules/OddModal/__tests__/SmallModalChildren.test.tsx +++ b/app/src/molecules/OddModal/__tests__/SmallModalChildren.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import '@testing-library/jest-dom/vitest' import { describe, it, expect, vi } from 'vitest' diff --git a/app/src/molecules/OffsetVector/__tests__/OffsetVector.test.tsx b/app/src/molecules/OffsetVector/__tests__/OffsetVector.test.tsx index 14dfd05db24..bb247c0175a 100644 --- a/app/src/molecules/OffsetVector/__tests__/OffsetVector.test.tsx +++ b/app/src/molecules/OffsetVector/__tests__/OffsetVector.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import '@testing-library/jest-dom/vitest' import { describe, it, expect, beforeEach } from 'vitest' diff --git a/app/src/molecules/OffsetVector/index.tsx b/app/src/molecules/OffsetVector/index.tsx index 47d4cf639d6..155019e8074 100644 --- a/app/src/molecules/OffsetVector/index.tsx +++ b/app/src/molecules/OffsetVector/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Flex, SPACING, diff --git a/app/src/molecules/PipetteSelect/index.tsx b/app/src/molecules/PipetteSelect/index.tsx index 80170fd0bc8..aec4e45300f 100644 --- a/app/src/molecules/PipetteSelect/index.tsx +++ b/app/src/molecules/PipetteSelect/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import groupBy from 'lodash/groupBy' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/molecules/PythonLabwareOffsetSnippet/createSnippet.ts b/app/src/molecules/PythonLabwareOffsetSnippet/createSnippet.ts index 4b6a1d4c236..79a5a70fad2 100644 --- a/app/src/molecules/PythonLabwareOffsetSnippet/createSnippet.ts +++ b/app/src/molecules/PythonLabwareOffsetSnippet/createSnippet.ts @@ -1,7 +1,7 @@ import isEqual from 'lodash/isEqual' import { getLoadedLabwareDefinitionsByUri } from '@opentrons/shared-data' -import { getLabwareDefinitionUri } from '/app/organisms/Devices/ProtocolRun/utils/getLabwareDefinitionUri' -import { getLabwareOffsetLocation } from '/app/organisms/Devices/ProtocolRun/utils/getLabwareOffsetLocation' +import { getLabwareDefinitionUri } from '/app/transformations/protocols' +import { getLabwareOffsetLocation } from '/app/transformations/analysis' import type { LabwareOffset } from '@opentrons/api-client' import type { LoadedLabware, diff --git a/app/src/molecules/PythonLabwareOffsetSnippet/index.tsx b/app/src/molecules/PythonLabwareOffsetSnippet/index.tsx index 00862359a37..d6e1e4bbd7d 100644 --- a/app/src/molecules/PythonLabwareOffsetSnippet/index.tsx +++ b/app/src/molecules/PythonLabwareOffsetSnippet/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import styled from 'styled-components' import { TYPOGRAPHY, SPACING, BORDERS, COLORS } from '@opentrons/components' import { createSnippet } from './createSnippet' @@ -32,8 +32,8 @@ export function PythonLabwareOffsetSnippet( props: PythonLabwareOffsetSnippetProps ): JSX.Element | null { const { commands, labware, modules, labwareOffsets, mode } = props - const [snippet, setSnippet] = React.useState(null) - React.useEffect(() => { + const [snippet, setSnippet] = useState(null) + useEffect(() => { if (labware.length > 0 && labwareOffsets != null) { setSnippet( createSnippet(mode, commands, labware, modules, labwareOffsets) diff --git a/app/src/molecules/ReleaseNotes/index.tsx b/app/src/molecules/ReleaseNotes/index.tsx index 5e59ecb5019..0c6207f84a1 100644 --- a/app/src/molecules/ReleaseNotes/index.tsx +++ b/app/src/molecules/ReleaseNotes/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import Markdown from 'react-markdown' import { Box, COLORS, SPACING, LegacyStyledText } from '@opentrons/components' diff --git a/app/src/organisms/Devices/ProtocolRun/RunTimer.tsx b/app/src/molecules/RunTimer/RunTimer.tsx similarity index 92% rename from app/src/organisms/Devices/ProtocolRun/RunTimer.tsx rename to app/src/molecules/RunTimer/RunTimer.tsx index ba285e0bb12..74962c82bb5 100644 --- a/app/src/organisms/Devices/ProtocolRun/RunTimer.tsx +++ b/app/src/molecules/RunTimer/RunTimer.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { RUN_STATUS_STOP_REQUESTED } from '@opentrons/api-client' import { @@ -24,7 +24,7 @@ export function RunTimer({ completedAt: string | null style?: CSSProp }): JSX.Element { - const [now, setNow] = React.useState(Date()) + const [now, setNow] = useState(Date()) useInterval( () => { setNow(Date()) diff --git a/app/src/molecules/RunTimer/index.ts b/app/src/molecules/RunTimer/index.ts new file mode 100644 index 00000000000..c48cfaa132c --- /dev/null +++ b/app/src/molecules/RunTimer/index.ts @@ -0,0 +1 @@ +export * from './RunTimer' diff --git a/app/src/molecules/SimpleWizardBody/SimpleWizardBody.stories.tsx b/app/src/molecules/SimpleWizardBody/SimpleWizardBody.stories.tsx index 7b66034657b..7e73f357d45 100644 --- a/app/src/molecules/SimpleWizardBody/SimpleWizardBody.stories.tsx +++ b/app/src/molecules/SimpleWizardBody/SimpleWizardBody.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Provider } from 'react-redux' import { createStore } from 'redux' import { COLORS, PrimaryButton, ModalShell } from '@opentrons/components' diff --git a/app/src/molecules/SimpleWizardBody/SimpleWizardBodyContainer.tsx b/app/src/molecules/SimpleWizardBody/SimpleWizardBodyContainer.tsx index 11644dba212..fef84670b19 100644 --- a/app/src/molecules/SimpleWizardBody/SimpleWizardBodyContainer.tsx +++ b/app/src/molecules/SimpleWizardBody/SimpleWizardBodyContainer.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { diff --git a/app/src/molecules/SimpleWizardBody/SimpleWizardBodyContent.tsx b/app/src/molecules/SimpleWizardBody/SimpleWizardBodyContent.tsx index 84e83966f07..16113e28c9a 100644 --- a/app/src/molecules/SimpleWizardBody/SimpleWizardBodyContent.tsx +++ b/app/src/molecules/SimpleWizardBody/SimpleWizardBodyContent.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useSelector } from 'react-redux' import { css } from 'styled-components' import { diff --git a/app/src/molecules/SimpleWizardBody/SimpleWizardInProgressBody.tsx b/app/src/molecules/SimpleWizardBody/SimpleWizardInProgressBody.tsx index 55bd83b534b..10882025dfc 100644 --- a/app/src/molecules/SimpleWizardBody/SimpleWizardInProgressBody.tsx +++ b/app/src/molecules/SimpleWizardBody/SimpleWizardInProgressBody.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import type { StyleProps } from '@opentrons/components' import { InProgressModal } from '../InProgressModal/InProgressModal' import { SimpleWizardBodyContainer } from './SimpleWizardBodyContainer' diff --git a/app/src/molecules/SimpleWizardBody/__tests__/SimpleWizardBody.test.tsx b/app/src/molecules/SimpleWizardBody/__tests__/SimpleWizardBody.test.tsx index 3d2b0103578..9849e8fa03c 100644 --- a/app/src/molecules/SimpleWizardBody/__tests__/SimpleWizardBody.test.tsx +++ b/app/src/molecules/SimpleWizardBody/__tests__/SimpleWizardBody.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import { COLORS } from '@opentrons/components' diff --git a/app/src/molecules/SimpleWizardBody/index.tsx b/app/src/molecules/SimpleWizardBody/index.tsx index c0408417030..b554b71b90e 100644 --- a/app/src/molecules/SimpleWizardBody/index.tsx +++ b/app/src/molecules/SimpleWizardBody/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { SimpleWizardBodyContainer } from './SimpleWizardBodyContainer' import { SimpleWizardBodyContent } from './SimpleWizardBodyContent' diff --git a/app/src/molecules/ToggleGroup/__tests__/useToggleGroup.test.tsx b/app/src/molecules/ToggleGroup/__tests__/useToggleGroup.test.tsx index 5312b7d33c9..c4829b79615 100644 --- a/app/src/molecules/ToggleGroup/__tests__/useToggleGroup.test.tsx +++ b/app/src/molecules/ToggleGroup/__tests__/useToggleGroup.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Provider } from 'react-redux' import { createStore } from 'redux' import '@testing-library/jest-dom/vitest' diff --git a/app/src/molecules/UnorderedList/index.tsx b/app/src/molecules/UnorderedList/index.tsx index 685adc170f5..cf9937266a8 100644 --- a/app/src/molecules/UnorderedList/index.tsx +++ b/app/src/molecules/UnorderedList/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { SPACING, LegacyStyledText } from '@opentrons/components' diff --git a/app/src/molecules/UpdateBanner/__tests__/UpdateBanner.test.tsx b/app/src/molecules/UpdateBanner/__tests__/UpdateBanner.test.tsx index c40d1e2e197..52b074b37d8 100644 --- a/app/src/molecules/UpdateBanner/__tests__/UpdateBanner.test.tsx +++ b/app/src/molecules/UpdateBanner/__tests__/UpdateBanner.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { when } from 'vitest-when' diff --git a/app/src/molecules/UpdateBanner/index.tsx b/app/src/molecules/UpdateBanner/index.tsx index 12927dc5eea..31a7dbe3d49 100644 --- a/app/src/molecules/UpdateBanner/index.tsx +++ b/app/src/molecules/UpdateBanner/index.tsx @@ -1,9 +1,9 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { ALIGN_START, Btn, + Banner, DIRECTION_COLUMN, Flex, SPACING, @@ -11,9 +11,8 @@ import { TYPOGRAPHY, useHoverTooltip, } from '@opentrons/components' - -import { Banner } from '/app/atoms/Banner' import { useIsFlex } from '/app/redux-resources/robots' + import { useIsEstopNotDisengaged } from '/app/resources/devices/hooks/useIsEstopNotDisengaged' interface UpdateBannerProps { diff --git a/app/src/molecules/UploadInput/UploadInput.stories.tsx b/app/src/molecules/UploadInput/UploadInput.stories.tsx index d27cb0198d8..461e23fd01a 100644 --- a/app/src/molecules/UploadInput/UploadInput.stories.tsx +++ b/app/src/molecules/UploadInput/UploadInput.stories.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - import { ALIGN_CENTER, COLORS, diff --git a/app/src/molecules/UploadInput/__tests__/UploadInput.test.tsx b/app/src/molecules/UploadInput/__tests__/UploadInput.test.tsx index 92e69095ba2..5c99328ade3 100644 --- a/app/src/molecules/UploadInput/__tests__/UploadInput.test.tsx +++ b/app/src/molecules/UploadInput/__tests__/UploadInput.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/molecules/WizardHeader/WizardHeader.stories.tsx b/app/src/molecules/WizardHeader/WizardHeader.stories.tsx index 70db49aa835..0d26e4f967d 100644 --- a/app/src/molecules/WizardHeader/WizardHeader.stories.tsx +++ b/app/src/molecules/WizardHeader/WizardHeader.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { WizardHeader } from './index' import type { Story, Meta } from '@storybook/react' diff --git a/app/src/molecules/WizardHeader/__tests__/WizardHeader.test.tsx b/app/src/molecules/WizardHeader/__tests__/WizardHeader.test.tsx index 5cecec7d87d..0e11447cac2 100644 --- a/app/src/molecules/WizardHeader/__tests__/WizardHeader.test.tsx +++ b/app/src/molecules/WizardHeader/__tests__/WizardHeader.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/molecules/WizardHeader/index.tsx b/app/src/molecules/WizardHeader/index.tsx index 1f190e64915..1b0d23baa39 100644 --- a/app/src/molecules/WizardHeader/index.tsx +++ b/app/src/molecules/WizardHeader/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' import { diff --git a/app/src/molecules/WizardRequiredEquipmentList/index.tsx b/app/src/molecules/WizardRequiredEquipmentList/index.tsx index 7f46754bf6f..3fbcd48671b 100644 --- a/app/src/molecules/WizardRequiredEquipmentList/index.tsx +++ b/app/src/molecules/WizardRequiredEquipmentList/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' diff --git a/app/src/molecules/modals/BottomButtonBar.tsx b/app/src/molecules/modals/BottomButtonBar.tsx index 5abb73828aa..d5c202e943c 100644 --- a/app/src/molecules/modals/BottomButtonBar.tsx +++ b/app/src/molecules/modals/BottomButtonBar.tsx @@ -1,6 +1,6 @@ // bottom button bar for modals // TODO(mc, 2018-08-18): maybe make this the default AlertModal behavior -import * as React from 'react' +import type * as React from 'react' import cx from 'classnames' import { OutlineButton } from '@opentrons/components' diff --git a/app/src/molecules/modals/ErrorModal.tsx b/app/src/molecules/modals/ErrorModal.tsx index f4a273a5bdb..f5422e5b2e0 100644 --- a/app/src/molecules/modals/ErrorModal.tsx +++ b/app/src/molecules/modals/ErrorModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { createPortal } from 'react-dom' import { Link } from 'react-router-dom' import { AlertModal } from '@opentrons/components' diff --git a/app/src/molecules/modals/ScrollableAlertModal.tsx b/app/src/molecules/modals/ScrollableAlertModal.tsx index 32aae3def4f..c98846899b8 100644 --- a/app/src/molecules/modals/ScrollableAlertModal.tsx +++ b/app/src/molecules/modals/ScrollableAlertModal.tsx @@ -1,5 +1,5 @@ // AlertModal with vertical scrolling -import * as React from 'react' +import type * as React from 'react' import omit from 'lodash/omit' import { AlertModal } from '@opentrons/components' diff --git a/app/src/organisms/AdvancedSettings/AdditionalCustomLabwareSourceFolder.tsx b/app/src/organisms/AdvancedSettings/AdditionalCustomLabwareSourceFolder.tsx index 5dc07daa93e..57708f2854d 100644 --- a/app/src/organisms/AdvancedSettings/AdditionalCustomLabwareSourceFolder.tsx +++ b/app/src/organisms/AdvancedSettings/AdditionalCustomLabwareSourceFolder.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useDispatch, useSelector } from 'react-redux' diff --git a/app/src/organisms/AdvancedSettings/ClearUnavailableRobots.tsx b/app/src/organisms/AdvancedSettings/ClearUnavailableRobots.tsx index 0ed7a097e88..3493f8b767c 100644 --- a/app/src/organisms/AdvancedSettings/ClearUnavailableRobots.tsx +++ b/app/src/organisms/AdvancedSettings/ClearUnavailableRobots.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import { useDispatch, useSelector } from 'react-redux' diff --git a/app/src/organisms/AdvancedSettings/EnableDevTools.tsx b/app/src/organisms/AdvancedSettings/EnableDevTools.tsx index c117908913b..55d832027e9 100644 --- a/app/src/organisms/AdvancedSettings/EnableDevTools.tsx +++ b/app/src/organisms/AdvancedSettings/EnableDevTools.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useSelector, useDispatch } from 'react-redux' diff --git a/app/src/organisms/AdvancedSettings/OT2AdvancedSettings.tsx b/app/src/organisms/AdvancedSettings/OT2AdvancedSettings.tsx index b0cb39caa1f..7ab3ebb0bd0 100644 --- a/app/src/organisms/AdvancedSettings/OT2AdvancedSettings.tsx +++ b/app/src/organisms/AdvancedSettings/OT2AdvancedSettings.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useSelector, useDispatch } from 'react-redux' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' diff --git a/app/src/organisms/AdvancedSettings/OverridePathToPython.tsx b/app/src/organisms/AdvancedSettings/OverridePathToPython.tsx index e8fdd2f9b92..d0e5b7d6d93 100644 --- a/app/src/organisms/AdvancedSettings/OverridePathToPython.tsx +++ b/app/src/organisms/AdvancedSettings/OverridePathToPython.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { useSelector, useDispatch } from 'react-redux' diff --git a/app/src/organisms/AdvancedSettings/PreventRobotCaching.tsx b/app/src/organisms/AdvancedSettings/PreventRobotCaching.tsx index b0429e50ae9..0b65e78854e 100644 --- a/app/src/organisms/AdvancedSettings/PreventRobotCaching.tsx +++ b/app/src/organisms/AdvancedSettings/PreventRobotCaching.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { useDispatch, useSelector } from 'react-redux' diff --git a/app/src/organisms/AdvancedSettings/ShowHeaterShakerAttachmentModal.tsx b/app/src/organisms/AdvancedSettings/ShowHeaterShakerAttachmentModal.tsx index 2883a1aba44..578872b389b 100644 --- a/app/src/organisms/AdvancedSettings/ShowHeaterShakerAttachmentModal.tsx +++ b/app/src/organisms/AdvancedSettings/ShowHeaterShakerAttachmentModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useDispatch, useSelector } from 'react-redux' diff --git a/app/src/organisms/AdvancedSettings/ShowLabwareOffsetSnippets.tsx b/app/src/organisms/AdvancedSettings/ShowLabwareOffsetSnippets.tsx index b0ec158a017..61dc9a62788 100644 --- a/app/src/organisms/AdvancedSettings/ShowLabwareOffsetSnippets.tsx +++ b/app/src/organisms/AdvancedSettings/ShowLabwareOffsetSnippets.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useSelector, useDispatch } from 'react-redux' diff --git a/app/src/organisms/AdvancedSettings/U2EInformation.tsx b/app/src/organisms/AdvancedSettings/U2EInformation.tsx index cfd1c59f5f0..fb55e883dba 100644 --- a/app/src/organisms/AdvancedSettings/U2EInformation.tsx +++ b/app/src/organisms/AdvancedSettings/U2EInformation.tsx @@ -1,10 +1,10 @@ -import * as React from 'react' import { useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' import { ALIGN_CENTER, Box, + Banner, COLORS, DIRECTION_COLUMN, Flex, @@ -15,7 +15,6 @@ import { TYPOGRAPHY, } from '@opentrons/components' -import { Banner } from '/app/atoms/Banner' import { getU2EAdapterDevice, getU2EWindowsDriverStatus, diff --git a/app/src/organisms/AdvancedSettings/UpdatedChannel.tsx b/app/src/organisms/AdvancedSettings/UpdatedChannel.tsx index 10cce14e67e..aca0348cb7b 100644 --- a/app/src/organisms/AdvancedSettings/UpdatedChannel.tsx +++ b/app/src/organisms/AdvancedSettings/UpdatedChannel.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { useDispatch, useSelector } from 'react-redux' diff --git a/app/src/organisms/AdvancedSettings/__tests__/AdditionalCustomLabwareSourceFolder.test.tsx b/app/src/organisms/AdvancedSettings/__tests__/AdditionalCustomLabwareSourceFolder.test.tsx index 59789b1c109..48a365afa0b 100644 --- a/app/src/organisms/AdvancedSettings/__tests__/AdditionalCustomLabwareSourceFolder.test.tsx +++ b/app/src/organisms/AdvancedSettings/__tests__/AdditionalCustomLabwareSourceFolder.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { describe, it, expect, vi, beforeEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' import { i18n } from '/app/i18n' diff --git a/app/src/organisms/AdvancedSettings/__tests__/ClearUnavailableRobots.test.tsx b/app/src/organisms/AdvancedSettings/__tests__/ClearUnavailableRobots.test.tsx index 352f3516204..008a74ad734 100644 --- a/app/src/organisms/AdvancedSettings/__tests__/ClearUnavailableRobots.test.tsx +++ b/app/src/organisms/AdvancedSettings/__tests__/ClearUnavailableRobots.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen, fireEvent } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import { useConditionalConfirm } from '@opentrons/components' diff --git a/app/src/organisms/AdvancedSettings/__tests__/EnableDevTools.test.tsx b/app/src/organisms/AdvancedSettings/__tests__/EnableDevTools.test.tsx index 5605b324688..25014b0cedf 100644 --- a/app/src/organisms/AdvancedSettings/__tests__/EnableDevTools.test.tsx +++ b/app/src/organisms/AdvancedSettings/__tests__/EnableDevTools.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen, fireEvent } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/AdvancedSettings/__tests__/OT2AdvancedSettings.test.tsx b/app/src/organisms/AdvancedSettings/__tests__/OT2AdvancedSettings.test.tsx index 6cfc5acd81f..3cff50268bd 100644 --- a/app/src/organisms/AdvancedSettings/__tests__/OT2AdvancedSettings.test.tsx +++ b/app/src/organisms/AdvancedSettings/__tests__/OT2AdvancedSettings.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen, fireEvent } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/AdvancedSettings/__tests__/OverridePathToPython.test.tsx b/app/src/organisms/AdvancedSettings/__tests__/OverridePathToPython.test.tsx index c7f001c73f0..78574ff0eca 100644 --- a/app/src/organisms/AdvancedSettings/__tests__/OverridePathToPython.test.tsx +++ b/app/src/organisms/AdvancedSettings/__tests__/OverridePathToPython.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/AdvancedSettings/__tests__/PreventRobotCaching.test.tsx b/app/src/organisms/AdvancedSettings/__tests__/PreventRobotCaching.test.tsx index 3fe5de32faf..094a1ffac90 100644 --- a/app/src/organisms/AdvancedSettings/__tests__/PreventRobotCaching.test.tsx +++ b/app/src/organisms/AdvancedSettings/__tests__/PreventRobotCaching.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { when } from 'vitest-when' import { describe, it, expect, vi, beforeEach } from 'vitest' import { screen, fireEvent } from '@testing-library/react' diff --git a/app/src/organisms/AdvancedSettings/__tests__/ShowHeaterShakerAttachmentModal.test.tsx b/app/src/organisms/AdvancedSettings/__tests__/ShowHeaterShakerAttachmentModal.test.tsx index 623ca64c493..a91eec5495f 100644 --- a/app/src/organisms/AdvancedSettings/__tests__/ShowHeaterShakerAttachmentModal.test.tsx +++ b/app/src/organisms/AdvancedSettings/__tests__/ShowHeaterShakerAttachmentModal.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import { i18n } from '/app/i18n' diff --git a/app/src/organisms/AdvancedSettings/__tests__/ShowLabwareOffsetSnippets.test.tsx b/app/src/organisms/AdvancedSettings/__tests__/ShowLabwareOffsetSnippets.test.tsx index 30eade7b245..047e310c628 100644 --- a/app/src/organisms/AdvancedSettings/__tests__/ShowLabwareOffsetSnippets.test.tsx +++ b/app/src/organisms/AdvancedSettings/__tests__/ShowLabwareOffsetSnippets.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import { i18n } from '/app/i18n' diff --git a/app/src/organisms/AdvancedSettings/__tests__/U2EInformation.test.tsx b/app/src/organisms/AdvancedSettings/__tests__/U2EInformation.test.tsx index 2aa4a5a7858..a885226096d 100644 --- a/app/src/organisms/AdvancedSettings/__tests__/U2EInformation.test.tsx +++ b/app/src/organisms/AdvancedSettings/__tests__/U2EInformation.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import { i18n } from '/app/i18n' diff --git a/app/src/organisms/AdvancedSettings/__tests__/UpdatedChannel.test.tsx b/app/src/organisms/AdvancedSettings/__tests__/UpdatedChannel.test.tsx index 84c3bf76438..c6feb1c0c8d 100644 --- a/app/src/organisms/AdvancedSettings/__tests__/UpdatedChannel.test.tsx +++ b/app/src/organisms/AdvancedSettings/__tests__/UpdatedChannel.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, vi, beforeEach } from 'vitest' import { i18n } from '/app/i18n' diff --git a/app/src/organisms/Alerts/AlertsModal.tsx b/app/src/organisms/Alerts/AlertsModal.tsx index 2e9375f4ec8..820756f907a 100644 --- a/app/src/organisms/Alerts/AlertsModal.tsx +++ b/app/src/organisms/Alerts/AlertsModal.tsx @@ -1,20 +1,20 @@ -import * as React from 'react' +import type { MutableRefObject } from 'react' +import { useState, useEffect } from 'react' import { useSelector, useDispatch } from 'react-redux' import { useTranslation } from 'react-i18next' import head from 'lodash/head' -import { SUCCESS_TOAST, WARNING_TOAST } from '@opentrons/components' +import { SUCCESS_TOAST, WARNING_TOAST } from '@opentrons/components' import * as AppAlerts from '/app/redux/alerts' import { getHasJustUpdated, toggleConfigValue } from '/app/redux/config' import { getAvailableShellUpdate } from '/app/redux/shell' import { useToaster } from '../ToasterOven' import { UpdateAppModal } from '../UpdateAppModal' import { U2EDriverOutdatedAlert } from './U2EDriverOutdatedAlert' -import { useRemoveActiveAppUpdateToast } from '.' +import { useRemoveActiveAppUpdateToast } from '.' import type { State, Dispatch } from '/app/redux/types' import type { AlertId } from '/app/redux/alerts/types' -import type { MutableRefObject } from 'react' interface AlertsModalProps { toastIdRef: MutableRefObject @@ -22,7 +22,7 @@ interface AlertsModalProps { export function AlertsModal({ toastIdRef }: AlertsModalProps): JSX.Element { const dispatch = useDispatch() - const [showUpdateModal, setShowUpdateModal] = React.useState(false) + const [showUpdateModal, setShowUpdateModal] = useState(false) const { t } = useTranslation(['app_settings', 'branded']) const { makeToast } = useToaster() const { removeActiveAppUpdateToast } = useRemoveActiveAppUpdateToast() @@ -51,7 +51,7 @@ export function AlertsModal({ toastIdRef }: AlertsModalProps): JSX.Element { isAppUpdateAvailable && !isAppUpdateIgnored // Only run this hook on app startup - React.useEffect(() => { + useEffect(() => { if (hasJustUpdated) { makeToast( t('branded:opentrons_app_successfully_updated') as string, @@ -65,7 +65,7 @@ export function AlertsModal({ toastIdRef }: AlertsModalProps): JSX.Element { } }, []) - React.useEffect(() => { + useEffect(() => { if (createAppUpdateAvailableToast) { toastIdRef.current = makeToast( t('branded:opentrons_app_update_available_variation') as string, diff --git a/app/src/organisms/Alerts/U2EDriverOutdatedAlert.tsx b/app/src/organisms/Alerts/U2EDriverOutdatedAlert.tsx index 5888fa19396..fbb79a6b935 100644 --- a/app/src/organisms/Alerts/U2EDriverOutdatedAlert.tsx +++ b/app/src/organisms/Alerts/U2EDriverOutdatedAlert.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Link as InternalLink } from 'react-router-dom' import styled from 'styled-components' diff --git a/app/src/organisms/Alerts/useRemoveActiveAppUpdateToast.ts.ts b/app/src/organisms/Alerts/useRemoveActiveAppUpdateToast.ts.ts index 030639ededa..a45a88716e8 100644 --- a/app/src/organisms/Alerts/useRemoveActiveAppUpdateToast.ts.ts +++ b/app/src/organisms/Alerts/useRemoveActiveAppUpdateToast.ts.ts @@ -1,7 +1,7 @@ -import * as React from 'react' +import { useContext } from 'react' import { AlertsContext } from '.' import type { AlertsContextProps } from '.' export function useRemoveActiveAppUpdateToast(): AlertsContextProps { - return React.useContext(AlertsContext) + return useContext(AlertsContext) } diff --git a/app/src/organisms/AppSettings/ConnectRobotSlideout.tsx b/app/src/organisms/AppSettings/ConnectRobotSlideout.tsx index 4e003d3f9cf..7c3148796d9 100644 --- a/app/src/organisms/AppSettings/ConnectRobotSlideout.tsx +++ b/app/src/organisms/AppSettings/ConnectRobotSlideout.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { useSelector, useDispatch } from 'react-redux' import { useTranslation } from 'react-i18next' @@ -37,10 +37,10 @@ export function ConnectRobotSlideout({ isExpanded, onCloseClick, }: ConnectRobotSlideoutProps): JSX.Element | null { - const [mostRecentAddition, setMostRecentAddition] = React.useState< - string | null - >(null) - const [mostRecentDiscovered, setMostRecentDiscovered] = React.useState< + const [mostRecentAddition, setMostRecentAddition] = useState( + null + ) + const [mostRecentDiscovered, setMostRecentDiscovered] = useState< boolean | null >(null) const { t } = useTranslation(['app_settings', 'shared', 'branded']) @@ -62,7 +62,7 @@ export function ConnectRobotSlideout({ ) } - React.useEffect(() => { + useEffect(() => { dispatch(startDiscovery()) }, [dispatch]) diff --git a/app/src/organisms/AppSettings/FeatureFlags.tsx b/app/src/organisms/AppSettings/FeatureFlags.tsx index f53c5501056..d86b52534ab 100644 --- a/app/src/organisms/AppSettings/FeatureFlags.tsx +++ b/app/src/organisms/AppSettings/FeatureFlags.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { Fragment } from 'react' import { useTranslation } from 'react-i18next' import { useSelector, useDispatch } from 'react-redux' import { @@ -32,7 +32,7 @@ export function FeatureFlags(): JSX.Element { paddingY={SPACING.spacing24} > {Config.DEV_INTERNAL_FLAGS.map((flag, index) => ( - + )} - + ))} ) diff --git a/app/src/organisms/AppSettings/ManualIpHostnameField.tsx b/app/src/organisms/AppSettings/ManualIpHostnameField.tsx index 8d249518ae4..c7b179ab4cd 100644 --- a/app/src/organisms/AppSettings/ManualIpHostnameField.tsx +++ b/app/src/organisms/AppSettings/ManualIpHostnameField.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/AppSettings/ManualIpHostnameForm.tsx b/app/src/organisms/AppSettings/ManualIpHostnameForm.tsx index ab9119c7fcf..52892422dce 100644 --- a/app/src/organisms/AppSettings/ManualIpHostnameForm.tsx +++ b/app/src/organisms/AppSettings/ManualIpHostnameForm.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useDispatch } from 'react-redux' import { useTranslation } from 'react-i18next' import { useForm } from 'react-hook-form' diff --git a/app/src/organisms/AppSettings/ManualIpHostnameItem.tsx b/app/src/organisms/AppSettings/ManualIpHostnameItem.tsx index 70dc5797d64..5a1dde3b663 100644 --- a/app/src/organisms/AppSettings/ManualIpHostnameItem.tsx +++ b/app/src/organisms/AppSettings/ManualIpHostnameItem.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useEffect } from 'react' import { useTranslation } from 'react-i18next' import styled, { css } from 'styled-components' @@ -68,7 +68,7 @@ export function ManualIpHostnameItem({ } } - React.useEffect(() => { + useEffect(() => { if (justAdded) { setMostRecentDiscovered(discovered) // Note this is to avoid the case that not found but not display the message diff --git a/app/src/organisms/AppSettings/ManualIpHostnameList.tsx b/app/src/organisms/AppSettings/ManualIpHostnameList.tsx index f74675750ca..1e96b60010d 100644 --- a/app/src/organisms/AppSettings/ManualIpHostnameList.tsx +++ b/app/src/organisms/AppSettings/ManualIpHostnameList.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { Fragment } from 'react' import { useSelector, useDispatch } from 'react-redux' import { getConfig, removeManualIp } from '/app/redux/config' @@ -38,7 +38,7 @@ export function ManualIpHostnameList({ bDiscovered && !aDiscovered ? -1 : 1 ) .map(([candidate, discovered], index) => ( - + dispatch(removeManualIp(candidate))} @@ -48,7 +48,7 @@ export function ManualIpHostnameList({ setMostRecentDiscovered={setMostRecentDiscovered} isLast={index === candidates.length - 1} /> - + )) : null} diff --git a/app/src/organisms/AppSettings/PreviousVersionModal.tsx b/app/src/organisms/AppSettings/PreviousVersionModal.tsx index 81fece0fbdc..3b3a58f2cc1 100644 --- a/app/src/organisms/AppSettings/PreviousVersionModal.tsx +++ b/app/src/organisms/AppSettings/PreviousVersionModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { Box, diff --git a/app/src/organisms/AppSettings/__tests__/ConnectRobotSlideout.test.tsx b/app/src/organisms/AppSettings/__tests__/ConnectRobotSlideout.test.tsx index 069a66dde38..c92e1f495a0 100644 --- a/app/src/organisms/AppSettings/__tests__/ConnectRobotSlideout.test.tsx +++ b/app/src/organisms/AppSettings/__tests__/ConnectRobotSlideout.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import '@testing-library/jest-dom/vitest' import { describe, it, expect, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/AppSettings/__tests__/PreviousVersionModal.test.tsx b/app/src/organisms/AppSettings/__tests__/PreviousVersionModal.test.tsx index 853bf6c2257..b4449623c05 100644 --- a/app/src/organisms/AppSettings/__tests__/PreviousVersionModal.test.tsx +++ b/app/src/organisms/AppSettings/__tests__/PreviousVersionModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/ApplyHistoricOffsets/LabwareOffsetTable.tsx b/app/src/organisms/ApplyHistoricOffsets/LabwareOffsetTable.tsx index 7c5e6f53446..acfc27db683 100644 --- a/app/src/organisms/ApplyHistoricOffsets/LabwareOffsetTable.tsx +++ b/app/src/organisms/ApplyHistoricOffsets/LabwareOffsetTable.tsx @@ -1,9 +1,8 @@ -import * as React from 'react' import styled from 'styled-components' import { useTranslation } from 'react-i18next' import { SPACING, TYPOGRAPHY, COLORS } from '@opentrons/components' import { OffsetVector } from '/app/molecules/OffsetVector' -import { formatTimestamp } from '../Devices/utils' +import { formatTimestamp } from '/app/transformations/runs' import { getDisplayLocation } from '../LabwarePositionCheck/utils/getDisplayLocation' import type { LabwareDefinition2 } from '@opentrons/shared-data' import type { OffsetCandidate } from './hooks/useOffsetCandidatesForAnalysis' diff --git a/app/src/organisms/ApplyHistoricOffsets/__tests__/ApplyHistoricOffsets.test.tsx b/app/src/organisms/ApplyHistoricOffsets/__tests__/ApplyHistoricOffsets.test.tsx index 8a638843a68..1bc7e55abd0 100644 --- a/app/src/organisms/ApplyHistoricOffsets/__tests__/ApplyHistoricOffsets.test.tsx +++ b/app/src/organisms/ApplyHistoricOffsets/__tests__/ApplyHistoricOffsets.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi } from 'vitest' diff --git a/app/src/organisms/ApplyHistoricOffsets/__tests__/LabwareOffsetTable.test.tsx b/app/src/organisms/ApplyHistoricOffsets/__tests__/LabwareOffsetTable.test.tsx index 742a5f7f50b..8b527e239f1 100644 --- a/app/src/organisms/ApplyHistoricOffsets/__tests__/LabwareOffsetTable.test.tsx +++ b/app/src/organisms/ApplyHistoricOffsets/__tests__/LabwareOffsetTable.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, expect } from 'vitest' import { fixture96Plate, fixtureTiprackAdapter } from '@opentrons/shared-data' diff --git a/app/src/organisms/ApplyHistoricOffsets/hooks/__tests__/useHistoricRunDetails.test.tsx b/app/src/organisms/ApplyHistoricOffsets/hooks/__tests__/useHistoricRunDetails.test.tsx index 5fe40a9333e..a697fabb2d3 100644 --- a/app/src/organisms/ApplyHistoricOffsets/hooks/__tests__/useHistoricRunDetails.test.tsx +++ b/app/src/organisms/ApplyHistoricOffsets/hooks/__tests__/useHistoricRunDetails.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, vi } from 'vitest' import { when } from 'vitest-when' import { renderHook, waitFor } from '@testing-library/react' diff --git a/app/src/organisms/ApplyHistoricOffsets/hooks/__tests__/useOffsetCandidatesForAnalysis.test.tsx b/app/src/organisms/ApplyHistoricOffsets/hooks/__tests__/useOffsetCandidatesForAnalysis.test.tsx index 48cfcfb3f78..832417cb9af 100644 --- a/app/src/organisms/ApplyHistoricOffsets/hooks/__tests__/useOffsetCandidatesForAnalysis.test.tsx +++ b/app/src/organisms/ApplyHistoricOffsets/hooks/__tests__/useOffsetCandidatesForAnalysis.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, vi, beforeEach } from 'vitest' import { when } from 'vitest-when' import { renderHook, waitFor } from '@testing-library/react' diff --git a/app/src/organisms/Breadcrumbs/__tests__/Breadcrumbs.test.tsx b/app/src/organisms/Breadcrumbs/__tests__/Breadcrumbs.test.tsx index 22216e8b830..559c6e3cf2b 100644 --- a/app/src/organisms/Breadcrumbs/__tests__/Breadcrumbs.test.tsx +++ b/app/src/organisms/Breadcrumbs/__tests__/Breadcrumbs.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter, Route, Routes } from 'react-router-dom' import { when } from 'vitest-when' import { describe, it, expect, beforeEach, vi } from 'vitest' @@ -6,7 +5,7 @@ import { describe, it, expect, beforeEach, vi } from 'vitest' import { fireEvent, screen } from '@testing-library/react' import { i18n } from '/app/i18n' -import { useRunCreatedAtTimestamp } from '/app/organisms/Devices/hooks' +import { useRunCreatedAtTimestamp } from '/app/resources/runs' import { useRobot } from '/app/redux-resources/robots' import { getProtocolDisplayName } from '/app/organisms/ProtocolsLanding/utils' import { getIsOnDevice } from '/app/redux/config' @@ -18,7 +17,7 @@ import { Breadcrumbs } from '..' import type { State } from '/app/redux/types' -vi.mock('/app/organisms/Devices/hooks') +vi.mock('/app/resources/runs') vi.mock('/app/redux-resources/robots') vi.mock('/app/organisms/ProtocolsLanding/utils') vi.mock('/app/redux/config') diff --git a/app/src/organisms/Breadcrumbs/index.tsx b/app/src/organisms/Breadcrumbs/index.tsx index f67493c2998..6b26b390f79 100644 --- a/app/src/organisms/Breadcrumbs/index.tsx +++ b/app/src/organisms/Breadcrumbs/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' import { Link, useParams, useLocation } from 'react-router-dom' @@ -18,7 +17,7 @@ import { } from '@opentrons/components' import { ApiHostProvider } from '@opentrons/react-api-client' -import { useRunCreatedAtTimestamp } from '/app/organisms/Devices/hooks' +import { useRunCreatedAtTimestamp } from '/app/resources/runs' import { getProtocolDisplayName } from '/app/organisms/ProtocolsLanding/utils' import { getIsOnDevice } from '/app/redux/config' import { OPENTRONS_USB } from '/app/redux/discovery' diff --git a/app/src/organisms/CalibrateDeck/__tests__/CalibrateDeck.test.tsx b/app/src/organisms/CalibrateDeck/__tests__/CalibrateDeck.test.tsx index fe899c601fd..ac5e8822416 100644 --- a/app/src/organisms/CalibrateDeck/__tests__/CalibrateDeck.test.tsx +++ b/app/src/organisms/CalibrateDeck/__tests__/CalibrateDeck.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, describe, beforeEach, expect, it } from 'vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/organisms/CalibratePipetteOffset/__tests__/CalibratePipetteOffset.test.tsx b/app/src/organisms/CalibratePipetteOffset/__tests__/CalibratePipetteOffset.test.tsx index a5eaae716ff..7f85d0ad109 100644 --- a/app/src/organisms/CalibratePipetteOffset/__tests__/CalibratePipetteOffset.test.tsx +++ b/app/src/organisms/CalibratePipetteOffset/__tests__/CalibratePipetteOffset.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, describe, expect, beforeEach } from 'vitest' import { when } from 'vitest-when' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/organisms/CalibratePipetteOffset/useCalibratePipetteOffset.tsx b/app/src/organisms/CalibratePipetteOffset/useCalibratePipetteOffset.tsx index 2759c30021d..2cbb250d73c 100644 --- a/app/src/organisms/CalibratePipetteOffset/useCalibratePipetteOffset.tsx +++ b/app/src/organisms/CalibratePipetteOffset/useCalibratePipetteOffset.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useRef, useEffect } from 'react' import { createPortal } from 'react-dom' import { useSelector, useDispatch } from 'react-redux' @@ -38,10 +38,10 @@ export function useCalibratePipetteOffset( onComplete: (() => unknown) | null = null ): [Invoker, JSX.Element | null] { const { t } = useTranslation(['robot_calibration', 'shared']) - const createRequestId = React.useRef(null) - const deleteRequestId = React.useRef(null) - const jogRequestId = React.useRef(null) - const spinnerRequestId = React.useRef(null) + const createRequestId = useRef(null) + const deleteRequestId = useRef(null) + const jogRequestId = useRef(null) + const spinnerRequestId = useRef(null) const dispatch = useDispatch() const pipOffsetCalSession: PipetteOffsetCalibrationSession | null = useSelector( @@ -112,7 +112,7 @@ export function useCalibratePipetteOffset( : null )?.status === RobotApi.PENDING - React.useEffect(() => { + useEffect(() => { if (shouldClose) { onComplete?.() deleteRequestId.current = null diff --git a/app/src/organisms/CalibrateTipLength/ConfirmRecalibrationModal.tsx b/app/src/organisms/CalibrateTipLength/ConfirmRecalibrationModal.tsx index fb413eed6cb..734b8f7ef0e 100644 --- a/app/src/organisms/CalibrateTipLength/ConfirmRecalibrationModal.tsx +++ b/app/src/organisms/CalibrateTipLength/ConfirmRecalibrationModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { createPortal } from 'react-dom' import { diff --git a/app/src/organisms/CalibrateTipLength/TipLengthCalibrationInfoBox.tsx b/app/src/organisms/CalibrateTipLength/TipLengthCalibrationInfoBox.tsx index 0869293922c..7bcf3bd81e3 100644 --- a/app/src/organisms/CalibrateTipLength/TipLengthCalibrationInfoBox.tsx +++ b/app/src/organisms/CalibrateTipLength/TipLengthCalibrationInfoBox.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Box, Text, diff --git a/app/src/organisms/CalibrateTipLength/__tests__/AskForCalibrationBlockModal.test.tsx b/app/src/organisms/CalibrateTipLength/__tests__/AskForCalibrationBlockModal.test.tsx index 5c9e4fac976..199ec279988 100644 --- a/app/src/organisms/CalibrateTipLength/__tests__/AskForCalibrationBlockModal.test.tsx +++ b/app/src/organisms/CalibrateTipLength/__tests__/AskForCalibrationBlockModal.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, it, describe, expect } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/CalibrateTipLength/__tests__/CalibrateTipLength.test.tsx b/app/src/organisms/CalibrateTipLength/__tests__/CalibrateTipLength.test.tsx index babf1c2e9d0..8753a346cbf 100644 --- a/app/src/organisms/CalibrateTipLength/__tests__/CalibrateTipLength.test.tsx +++ b/app/src/organisms/CalibrateTipLength/__tests__/CalibrateTipLength.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect, beforeEach, afterEach } from 'vitest' import { when } from 'vitest-when' diff --git a/app/src/organisms/CalibrationPanels/CalibrationLabwareRender.tsx b/app/src/organisms/CalibrationPanels/CalibrationLabwareRender.tsx index e81adfe525b..2e789e7f116 100644 --- a/app/src/organisms/CalibrationPanels/CalibrationLabwareRender.tsx +++ b/app/src/organisms/CalibrationPanels/CalibrationLabwareRender.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - import { LabwareRender, LabwareNameOverlay, diff --git a/app/src/organisms/CalibrationPanels/ChooseTipRack.tsx b/app/src/organisms/CalibrationPanels/ChooseTipRack.tsx index 7a09addeb70..9eba4813786 100644 --- a/app/src/organisms/CalibrationPanels/ChooseTipRack.tsx +++ b/app/src/organisms/CalibrationPanels/ChooseTipRack.tsx @@ -1,10 +1,11 @@ -import * as React from 'react' +import { useState } from 'react' import { useSelector } from 'react-redux' import { Trans, useTranslation } from 'react-i18next' import head from 'lodash/head' import isEqual from 'lodash/isEqual' import { ALIGN_CENTER, + Banner, Box, COLORS, DIRECTION_COLUMN, @@ -25,7 +26,6 @@ import { getTipLengthForPipetteAndTiprack, } from '/app/redux/calibration/' import { Select } from '/app/atoms/SelectField/Select' -import { Banner } from '/app/atoms/Banner' import { Divider } from '/app/atoms/structure' import { NeedHelpLink } from './NeedHelpLink' import { ChosenTipRackRender } from './ChosenTipRackRender' @@ -153,7 +153,7 @@ export function ChooseTipRack(props: ChooseTipRackProps): JSX.Element { ] : [...opentronsTipRacksOptions] - const [selectedValue, setSelectedValue] = React.useState< + const [selectedValue, setSelectedValue] = useState< SingleValue | MultiValue >( chosenTipRack != null diff --git a/app/src/organisms/CalibrationPanels/ChosenTipRackRender.tsx b/app/src/organisms/CalibrationPanels/ChosenTipRackRender.tsx index 98d1ebb4499..780bcac2973 100644 --- a/app/src/organisms/CalibrationPanels/ChosenTipRackRender.tsx +++ b/app/src/organisms/CalibrationPanels/ChosenTipRackRender.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { ALIGN_CENTER, diff --git a/app/src/organisms/CalibrationPanels/CompleteConfirmation.tsx b/app/src/organisms/CalibrationPanels/CompleteConfirmation.tsx index 7d11b2243a9..cbd0e214cd1 100644 --- a/app/src/organisms/CalibrationPanels/CompleteConfirmation.tsx +++ b/app/src/organisms/CalibrationPanels/CompleteConfirmation.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { ALIGN_CENTER, diff --git a/app/src/organisms/CalibrationPanels/ConfirmCrashRecovery.tsx b/app/src/organisms/CalibrationPanels/ConfirmCrashRecovery.tsx index 43a55cabaa2..b2ac7b52179 100644 --- a/app/src/organisms/CalibrationPanels/ConfirmCrashRecovery.tsx +++ b/app/src/organisms/CalibrationPanels/ConfirmCrashRecovery.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/CalibrationPanels/ConfirmExit.tsx b/app/src/organisms/CalibrationPanels/ConfirmExit.tsx index 4d52a7424e9..36ae9c7d5b8 100644 --- a/app/src/organisms/CalibrationPanels/ConfirmExit.tsx +++ b/app/src/organisms/CalibrationPanels/ConfirmExit.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { AlertPrimaryButton, diff --git a/app/src/organisms/CalibrationPanels/DeckSetup.tsx b/app/src/organisms/CalibrationPanels/DeckSetup.tsx index 230b0456161..22d648a5335 100644 --- a/app/src/organisms/CalibrationPanels/DeckSetup.tsx +++ b/app/src/organisms/CalibrationPanels/DeckSetup.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useMemo } from 'react' import { useTranslation } from 'react-i18next' import map from 'lodash/map' @@ -28,7 +28,7 @@ import type { CalibrationPanelProps } from './types' const TIPRACK = 'tip rack' export function DeckSetup(props: CalibrationPanelProps): JSX.Element { - const deckDef = React.useMemo(() => getDeckDefinitions().ot2_standard, []) + const deckDef = useMemo(() => getDeckDefinitions().ot2_standard, []) const { t } = useTranslation('robot_calibration') diff --git a/app/src/organisms/CalibrationPanels/Introduction/Body.tsx b/app/src/organisms/CalibrationPanels/Introduction/Body.tsx index 410e6d02423..8f67637f6c4 100644 --- a/app/src/organisms/CalibrationPanels/Introduction/Body.tsx +++ b/app/src/organisms/CalibrationPanels/Introduction/Body.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { LegacyStyledText } from '@opentrons/components' import * as Sessions from '/app/redux/sessions' diff --git a/app/src/organisms/CalibrationPanels/Introduction/InvalidationWarning.tsx b/app/src/organisms/CalibrationPanels/Introduction/InvalidationWarning.tsx index fd047275c46..dbe27f7358a 100644 --- a/app/src/organisms/CalibrationPanels/Introduction/InvalidationWarning.tsx +++ b/app/src/organisms/CalibrationPanels/Introduction/InvalidationWarning.tsx @@ -1,12 +1,11 @@ -import * as React from 'react' import { Flex, + Banner, SPACING, TYPOGRAPHY, LegacyStyledText, } from '@opentrons/components' import { useTranslation } from 'react-i18next' -import { Banner } from '/app/atoms/Banner' import * as Sessions from '/app/redux/sessions' interface InvalidationWarningProps { diff --git a/app/src/organisms/CalibrationPanels/Introduction/__tests__/Body.test.tsx b/app/src/organisms/CalibrationPanels/Introduction/__tests__/Body.test.tsx index 5de8c58d35c..c1300fb3218 100644 --- a/app/src/organisms/CalibrationPanels/Introduction/__tests__/Body.test.tsx +++ b/app/src/organisms/CalibrationPanels/Introduction/__tests__/Body.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { it, describe } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/CalibrationPanels/Introduction/__tests__/Introduction.test.tsx b/app/src/organisms/CalibrationPanels/Introduction/__tests__/Introduction.test.tsx index a5f0f6b59c0..a34460571ed 100644 --- a/app/src/organisms/CalibrationPanels/Introduction/__tests__/Introduction.test.tsx +++ b/app/src/organisms/CalibrationPanels/Introduction/__tests__/Introduction.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, expect, describe, beforeEach, afterEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/organisms/CalibrationPanels/Introduction/__tests__/InvalidationWarning.test.tsx b/app/src/organisms/CalibrationPanels/Introduction/__tests__/InvalidationWarning.test.tsx index da55d8a6b26..6151097eccf 100644 --- a/app/src/organisms/CalibrationPanels/Introduction/__tests__/InvalidationWarning.test.tsx +++ b/app/src/organisms/CalibrationPanels/Introduction/__tests__/InvalidationWarning.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { it, describe } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/CalibrationPanels/Introduction/index.tsx b/app/src/organisms/CalibrationPanels/Introduction/index.tsx index 1f5f85dbd16..3106786ec30 100644 --- a/app/src/organisms/CalibrationPanels/Introduction/index.tsx +++ b/app/src/organisms/CalibrationPanels/Introduction/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { getLabwareDisplayName } from '@opentrons/shared-data' import { @@ -39,11 +39,10 @@ export function Introduction(props: CalibrationPanelProps): JSX.Element { } = props const { t } = useTranslation('robot_calibration') - const [showChooseTipRack, setShowChooseTipRack] = React.useState(false) - const [ - chosenTipRack, - setChosenTipRack, - ] = React.useState(null) + const [showChooseTipRack, setShowChooseTipRack] = useState(false) + const [chosenTipRack, setChosenTipRack] = useState( + null + ) const handleChosenTipRack = (value: LabwareDefinition2 | null): void => { value != null && setChosenTipRack(value) diff --git a/app/src/organisms/CalibrationPanels/LoadingState.tsx b/app/src/organisms/CalibrationPanels/LoadingState.tsx index 03e491af459..0327e13d86c 100644 --- a/app/src/organisms/CalibrationPanels/LoadingState.tsx +++ b/app/src/organisms/CalibrationPanels/LoadingState.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { ALIGN_CENTER, COLORS, diff --git a/app/src/organisms/CalibrationPanels/MeasureNozzle.tsx b/app/src/organisms/CalibrationPanels/MeasureNozzle.tsx index 34afa5d7fd2..7e5e6f38f80 100644 --- a/app/src/organisms/CalibrationPanels/MeasureNozzle.tsx +++ b/app/src/organisms/CalibrationPanels/MeasureNozzle.tsx @@ -1,5 +1,5 @@ /* eslint-disable no-return-assign */ -import * as React from 'react' +import { useMemo } from 'react' import { css } from 'styled-components' import { useTranslation } from 'react-i18next' import { @@ -82,7 +82,7 @@ export function MeasureNozzle(props: CalibrationPanelProps): JSX.Element { const { t } = useTranslation('robot_calibration') const { sendCommands, calBlock, mount, isMulti, sessionType } = props - const demoAsset = React.useMemo( + const demoAsset = useMemo( () => calBlock != null ? assetMapBlock[sessionType]?.[mount]?.[isMulti ? 'multi' : 'single'] diff --git a/app/src/organisms/CalibrationPanels/MeasureTip.tsx b/app/src/organisms/CalibrationPanels/MeasureTip.tsx index 63bb5723bad..542cc1a1511 100644 --- a/app/src/organisms/CalibrationPanels/MeasureTip.tsx +++ b/app/src/organisms/CalibrationPanels/MeasureTip.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useMemo } from 'react' import { css } from 'styled-components' import { useTranslation } from 'react-i18next' import { @@ -82,7 +82,7 @@ export function MeasureTip(props: CalibrationPanelProps): JSX.Element { const { t } = useTranslation('robot_calibration') const { sendCommands, calBlock, isMulti, mount, sessionType } = props - const demoAsset = React.useMemo( + const demoAsset = useMemo( () => calBlock != null ? assetMapBlock[sessionType]?.[mount]?.[isMulti ? 'multi' : 'single'] diff --git a/app/src/organisms/CalibrationPanels/NeedHelpLink.tsx b/app/src/organisms/CalibrationPanels/NeedHelpLink.tsx index 01dac649b39..a87a7808827 100644 --- a/app/src/organisms/CalibrationPanels/NeedHelpLink.tsx +++ b/app/src/organisms/CalibrationPanels/NeedHelpLink.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { Flex, diff --git a/app/src/organisms/CalibrationPanels/SaveXYPoint.tsx b/app/src/organisms/CalibrationPanels/SaveXYPoint.tsx index 207f0a609c9..c727f9ce6fe 100644 --- a/app/src/organisms/CalibrationPanels/SaveXYPoint.tsx +++ b/app/src/organisms/CalibrationPanels/SaveXYPoint.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useMemo } from 'react' import { css } from 'styled-components' import { useTranslation } from 'react-i18next' import { @@ -143,7 +143,7 @@ export function SaveXYPoint(props: CalibrationPanelProps): JSX.Element | null { const { slotNumber, moveCommand } = contentsBySessionTypeByCurrentStep[sessionType]?.[currentStep] ?? {} - const demoAsset = React.useMemo( + const demoAsset = useMemo( () => slotNumber != null ? assetMap[slotNumber][mount][isMulti ? 'multi' : 'single'] diff --git a/app/src/organisms/CalibrationPanels/TipConfirmation.tsx b/app/src/organisms/CalibrationPanels/TipConfirmation.tsx index da25aef6ee5..88495309621 100644 --- a/app/src/organisms/CalibrationPanels/TipConfirmation.tsx +++ b/app/src/organisms/CalibrationPanels/TipConfirmation.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { DIRECTION_COLUMN, diff --git a/app/src/organisms/CalibrationPanels/TipPickUp.tsx b/app/src/organisms/CalibrationPanels/TipPickUp.tsx index f8008a98ede..d7360f9c052 100644 --- a/app/src/organisms/CalibrationPanels/TipPickUp.tsx +++ b/app/src/organisms/CalibrationPanels/TipPickUp.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { Trans, useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/CalibrationPanels/__tests__/ChooseTipRack.test.tsx b/app/src/organisms/CalibrationPanels/__tests__/ChooseTipRack.test.tsx index fb41e694dbd..4bb1cecd0ef 100644 --- a/app/src/organisms/CalibrationPanels/__tests__/ChooseTipRack.test.tsx +++ b/app/src/organisms/CalibrationPanels/__tests__/ChooseTipRack.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect, beforeEach } from 'vitest' diff --git a/app/src/organisms/CalibrationPanels/__tests__/ChosenTipRackRender.test.tsx b/app/src/organisms/CalibrationPanels/__tests__/ChosenTipRackRender.test.tsx index 11662e1bd1e..5eb7fa1db8b 100644 --- a/app/src/organisms/CalibrationPanels/__tests__/ChosenTipRackRender.test.tsx +++ b/app/src/organisms/CalibrationPanels/__tests__/ChosenTipRackRender.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { it, describe, beforeEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/CalibrationPanels/__tests__/CompleteConfirmation.test.tsx b/app/src/organisms/CalibrationPanels/__tests__/CompleteConfirmation.test.tsx index d5ff26ba96c..99dc73310d1 100644 --- a/app/src/organisms/CalibrationPanels/__tests__/CompleteConfirmation.test.tsx +++ b/app/src/organisms/CalibrationPanels/__tests__/CompleteConfirmation.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect } from 'vitest' diff --git a/app/src/organisms/CalibrationPanels/__tests__/ConfirmCrashRecovery.test.tsx b/app/src/organisms/CalibrationPanels/__tests__/ConfirmCrashRecovery.test.tsx index 51f70785f82..53fb8559e55 100644 --- a/app/src/organisms/CalibrationPanels/__tests__/ConfirmCrashRecovery.test.tsx +++ b/app/src/organisms/CalibrationPanels/__tests__/ConfirmCrashRecovery.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect } from 'vitest' diff --git a/app/src/organisms/CalibrationPanels/__tests__/ConfirmExit.test.tsx b/app/src/organisms/CalibrationPanels/__tests__/ConfirmExit.test.tsx index eaec356c62c..086c122e8bf 100644 --- a/app/src/organisms/CalibrationPanels/__tests__/ConfirmExit.test.tsx +++ b/app/src/organisms/CalibrationPanels/__tests__/ConfirmExit.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect } from 'vitest' diff --git a/app/src/organisms/CalibrationPanels/__tests__/DeckSetup.test.tsx b/app/src/organisms/CalibrationPanels/__tests__/DeckSetup.test.tsx index 920e6463cb9..bfa641e6b26 100644 --- a/app/src/organisms/CalibrationPanels/__tests__/DeckSetup.test.tsx +++ b/app/src/organisms/CalibrationPanels/__tests__/DeckSetup.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/CalibrationPanels/__tests__/MeasureNozzle.test.tsx b/app/src/organisms/CalibrationPanels/__tests__/MeasureNozzle.test.tsx index a7a8d68773f..d0de251b0ad 100644 --- a/app/src/organisms/CalibrationPanels/__tests__/MeasureNozzle.test.tsx +++ b/app/src/organisms/CalibrationPanels/__tests__/MeasureNozzle.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect } from 'vitest' diff --git a/app/src/organisms/CalibrationPanels/__tests__/MeasureTip.test.tsx b/app/src/organisms/CalibrationPanels/__tests__/MeasureTip.test.tsx index f2436f9af0f..70d8ab54a6b 100644 --- a/app/src/organisms/CalibrationPanels/__tests__/MeasureTip.test.tsx +++ b/app/src/organisms/CalibrationPanels/__tests__/MeasureTip.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect } from 'vitest' diff --git a/app/src/organisms/CalibrationPanels/__tests__/SaveXYPoint.test.tsx b/app/src/organisms/CalibrationPanels/__tests__/SaveXYPoint.test.tsx index feabd927785..3d7dbda32a9 100644 --- a/app/src/organisms/CalibrationPanels/__tests__/SaveXYPoint.test.tsx +++ b/app/src/organisms/CalibrationPanels/__tests__/SaveXYPoint.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect } from 'vitest' diff --git a/app/src/organisms/CalibrationPanels/__tests__/SaveZPoint.test.tsx b/app/src/organisms/CalibrationPanels/__tests__/SaveZPoint.test.tsx index 480686c489b..36f04c1ebb2 100644 --- a/app/src/organisms/CalibrationPanels/__tests__/SaveZPoint.test.tsx +++ b/app/src/organisms/CalibrationPanels/__tests__/SaveZPoint.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect } from 'vitest' diff --git a/app/src/organisms/CalibrationPanels/__tests__/TipConfirmation.test.tsx b/app/src/organisms/CalibrationPanels/__tests__/TipConfirmation.test.tsx index a0b26400005..f9eee22d78b 100644 --- a/app/src/organisms/CalibrationPanels/__tests__/TipConfirmation.test.tsx +++ b/app/src/organisms/CalibrationPanels/__tests__/TipConfirmation.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect } from 'vitest' diff --git a/app/src/organisms/CalibrationPanels/__tests__/TipPickUp.test.tsx b/app/src/organisms/CalibrationPanels/__tests__/TipPickUp.test.tsx index 7aa63cc6a2f..b6836cf96fb 100644 --- a/app/src/organisms/CalibrationPanels/__tests__/TipPickUp.test.tsx +++ b/app/src/organisms/CalibrationPanels/__tests__/TipPickUp.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect } from 'vitest' diff --git a/app/src/organisms/CalibrationPanels/__tests__/useConfirmCrashRecovery.test.tsx b/app/src/organisms/CalibrationPanels/__tests__/useConfirmCrashRecovery.test.tsx index 54ca33eb5b3..9e385a2c320 100644 --- a/app/src/organisms/CalibrationPanels/__tests__/useConfirmCrashRecovery.test.tsx +++ b/app/src/organisms/CalibrationPanels/__tests__/useConfirmCrashRecovery.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, renderHook, screen } from '@testing-library/react' import { I18nextProvider } from 'react-i18next' import { vi, it, describe, expect } from 'vitest' diff --git a/app/src/organisms/CalibrationPanels/useConfirmCrashRecovery.tsx b/app/src/organisms/CalibrationPanels/useConfirmCrashRecovery.tsx index 5f6058281c5..e4b9bb2c13f 100644 --- a/app/src/organisms/CalibrationPanels/useConfirmCrashRecovery.tsx +++ b/app/src/organisms/CalibrationPanels/useConfirmCrashRecovery.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { @@ -20,7 +20,7 @@ export function useConfirmCrashRecovery( ): [link: JSX.Element, confirmation: JSX.Element | null] { const { t } = useTranslation('robot_calibration') const { sendCommands } = props - const [showModal, setShowModal] = React.useState(false) + const [showModal, setShowModal] = useState(false) const doStartOver = (): void => { sendCommands({ command: Sessions.sharedCalCommands.INVALIDATE_LAST_ACTION }) diff --git a/app/src/organisms/CalibrationStatusCard/__tests__/CalibrationStatusCard.test.tsx b/app/src/organisms/CalibrationStatusCard/__tests__/CalibrationStatusCard.test.tsx index a1060e5200d..400e73b3078 100644 --- a/app/src/organisms/CalibrationStatusCard/__tests__/CalibrationStatusCard.test.tsx +++ b/app/src/organisms/CalibrationStatusCard/__tests__/CalibrationStatusCard.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { vi, it, describe, expect, beforeEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/organisms/CalibrationStatusCard/index.tsx b/app/src/organisms/CalibrationStatusCard/index.tsx index 057a9403585..9be7eecb1c2 100644 --- a/app/src/organisms/CalibrationStatusCard/index.tsx +++ b/app/src/organisms/CalibrationStatusCard/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { Link as RouterLink } from 'react-router-dom' diff --git a/app/src/organisms/CalibrationTaskList/__tests__/CalibrationTaskList.test.tsx b/app/src/organisms/CalibrationTaskList/__tests__/CalibrationTaskList.test.tsx index fe175420b4a..ebada58f1f3 100644 --- a/app/src/organisms/CalibrationTaskList/__tests__/CalibrationTaskList.test.tsx +++ b/app/src/organisms/CalibrationTaskList/__tests__/CalibrationTaskList.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { vi, it, describe, expect, beforeEach, afterEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' @@ -18,15 +17,14 @@ import { expectedIncompleteRightMountTaskList, expectedIncompleteLeftMountTaskList, } from '../../Devices/hooks/__fixtures__/taskListFixtures' -import { - useCalibrationTaskList, - useRunHasStarted, - useAttachedPipettes, -} from '../../Devices/hooks' +import { useCalibrationTaskList } from '../../Devices/hooks' +import { useAttachedPipettes } from '/app/resources/instruments' import { mockLeftProtoPipette } from '/app/redux/pipettes/__fixtures__' +import { useRunHasStarted } from '/app/resources/runs' vi.mock('../../Devices/hooks') vi.mock('/app/resources/runs') +vi.mock('/app/resources/instruments') const render = (robotName: string = 'otie') => { return renderWithProviders( diff --git a/app/src/organisms/CalibrationTaskList/index.tsx b/app/src/organisms/CalibrationTaskList/index.tsx index 069e6ba4ebf..8596e82e77c 100644 --- a/app/src/organisms/CalibrationTaskList/index.tsx +++ b/app/src/organisms/CalibrationTaskList/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useRef, useState, useEffect } from 'react' import { css } from 'styled-components' import { useTranslation } from 'react-i18next' import { useNavigate } from 'react-router-dom' @@ -20,12 +20,9 @@ import { import { StatusLabel } from '/app/atoms/StatusLabel' import { TaskList } from '../TaskList' -import { - useAttachedPipettes, - useCalibrationTaskList, - useRunHasStarted, -} from '/app/organisms/Devices/hooks' -import { useCurrentRunId } from '/app/resources/runs' +import { useCalibrationTaskList } from '/app/organisms/Devices/hooks' +import { useAttachedPipettes } from '/app/resources/instruments' +import { useCurrentRunId, useRunHasStarted } from '/app/resources/runs' import type { DashboardCalOffsetInvoker, @@ -48,14 +45,11 @@ export function CalibrationTaskList({ deckCalLauncher, exitBeforeDeckConfigCompletion, }: CalibrationTaskListProps): JSX.Element { - const prevActiveIndex = React.useRef<[number, number] | null>(null) - const [hasLaunchedWizard, setHasLaunchedWizard] = React.useState( + const prevActiveIndex = useRef<[number, number] | null>(null) + const [hasLaunchedWizard, setHasLaunchedWizard] = useState(false) + const [showCompletionScreen, setShowCompletionScreen] = useState( false ) - const [ - showCompletionScreen, - setShowCompletionScreen, - ] = React.useState(false) const { t } = useTranslation(['robot_calibration', 'device_settings']) const navigate = useNavigate() const { activeIndex, taskList, taskListStatus } = useCalibrationTaskList( @@ -80,7 +74,7 @@ export function CalibrationTaskList({ 'device_settings:some_robot_controls_are_not_available' ) - React.useEffect(() => { + useEffect(() => { if ( prevActiveIndex.current !== null && activeIndex === null && diff --git a/app/src/organisms/ChangePipette/ClearDeckModal.tsx b/app/src/organisms/ChangePipette/ClearDeckModal.tsx index dc89fc4ff7c..5fb2cd79de1 100644 --- a/app/src/organisms/ChangePipette/ClearDeckModal.tsx +++ b/app/src/organisms/ChangePipette/ClearDeckModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { DIRECTION_COLUMN, diff --git a/app/src/organisms/ChangePipette/ConfirmPipette.tsx b/app/src/organisms/ChangePipette/ConfirmPipette.tsx index c9c603905a1..f756059d003 100644 --- a/app/src/organisms/ChangePipette/ConfirmPipette.tsx +++ b/app/src/organisms/ChangePipette/ConfirmPipette.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { OT2_ROBOT_TYPE } from '@opentrons/shared-data' import { diff --git a/app/src/organisms/ChangePipette/ExitModal.tsx b/app/src/organisms/ChangePipette/ExitModal.tsx index 15ab8132dc6..619b35cea36 100644 --- a/app/src/organisms/ChangePipette/ExitModal.tsx +++ b/app/src/organisms/ChangePipette/ExitModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { COLORS, diff --git a/app/src/organisms/ChangePipette/InstructionStep.tsx b/app/src/organisms/ChangePipette/InstructionStep.tsx index a12ae6827e3..05d43fdd11c 100644 --- a/app/src/organisms/ChangePipette/InstructionStep.tsx +++ b/app/src/organisms/ChangePipette/InstructionStep.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Box, Flex, JUSTIFY_SPACE_EVENLY, SPACING } from '@opentrons/components' import type { PipetteChannels, diff --git a/app/src/organisms/ChangePipette/Instructions.tsx b/app/src/organisms/ChangePipette/Instructions.tsx index 4582dafafe5..cffdceca835 100644 --- a/app/src/organisms/ChangePipette/Instructions.tsx +++ b/app/src/organisms/ChangePipette/Instructions.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useEffect } from 'react' import { css } from 'styled-components' import { Trans, useTranslation } from 'react-i18next' import { @@ -69,7 +69,7 @@ export function Instructions(props: Props): JSX.Element { } = props const { t } = useTranslation('change_pipette') - React.useEffect(() => { + useEffect(() => { if (direction === 'detach' && currentStepCount === 0) { nextStep() } diff --git a/app/src/organisms/ChangePipette/LevelPipette.tsx b/app/src/organisms/ChangePipette/LevelPipette.tsx index d3f69909f5d..db49a4d6861 100644 --- a/app/src/organisms/ChangePipette/LevelPipette.tsx +++ b/app/src/organisms/ChangePipette/LevelPipette.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { css } from 'styled-components' diff --git a/app/src/organisms/ChangePipette/PipetteSelection.tsx b/app/src/organisms/ChangePipette/PipetteSelection.tsx index 58151b8610d..c306f132154 100644 --- a/app/src/organisms/ChangePipette/PipetteSelection.tsx +++ b/app/src/organisms/ChangePipette/PipetteSelection.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { DIRECTION_COLUMN, diff --git a/app/src/organisms/ChangePipette/__tests__/ChangePipette.test.tsx b/app/src/organisms/ChangePipette/__tests__/ChangePipette.test.tsx index f627265493c..89112a484d4 100644 --- a/app/src/organisms/ChangePipette/__tests__/ChangePipette.test.tsx +++ b/app/src/organisms/ChangePipette/__tests__/ChangePipette.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, describe, expect, beforeEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' @@ -15,7 +15,7 @@ import { SUCCESS, useDispatchApiRequests, } from '/app/redux/robot-api' -import { useAttachedPipettes } from '../../Devices/hooks' +import { useAttachedPipettes } from '/app/resources/instruments' import { PipetteSelection } from '../PipetteSelection' import { ExitModal } from '../ExitModal' import { ConfirmPipette } from '../ConfirmPipette' @@ -51,7 +51,7 @@ vi.mock('../PipetteSelection') vi.mock('../ExitModal') vi.mock('/app/molecules/InProgressModal/InProgressModal') vi.mock('../ConfirmPipette') -vi.mock('../../Devices/hooks') +vi.mock('/app/resources/instruments') vi.mock('/app/assets/images') const render = (props: React.ComponentProps) => { diff --git a/app/src/organisms/ChangePipette/__tests__/CheckPipettesButton.test.tsx b/app/src/organisms/ChangePipette/__tests__/CheckPipettesButton.test.tsx index 41c8afcb08c..a5fa3a50bd6 100644 --- a/app/src/organisms/ChangePipette/__tests__/CheckPipettesButton.test.tsx +++ b/app/src/organisms/ChangePipette/__tests__/CheckPipettesButton.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect, beforeEach } from 'vitest' diff --git a/app/src/organisms/ChangePipette/__tests__/ClearDeckModal.test.tsx b/app/src/organisms/ChangePipette/__tests__/ClearDeckModal.test.tsx index faae99c439c..f4af4566141 100644 --- a/app/src/organisms/ChangePipette/__tests__/ClearDeckModal.test.tsx +++ b/app/src/organisms/ChangePipette/__tests__/ClearDeckModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect, beforeEach } from 'vitest' diff --git a/app/src/organisms/ChangePipette/__tests__/ConfirmPipette.test.tsx b/app/src/organisms/ChangePipette/__tests__/ConfirmPipette.test.tsx index b2ddff2791f..a328df38383 100644 --- a/app/src/organisms/ChangePipette/__tests__/ConfirmPipette.test.tsx +++ b/app/src/organisms/ChangePipette/__tests__/ConfirmPipette.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect } from 'vitest' diff --git a/app/src/organisms/ChangePipette/__tests__/ExitModal.test.tsx b/app/src/organisms/ChangePipette/__tests__/ExitModal.test.tsx index f7438b1874c..29bf417071c 100644 --- a/app/src/organisms/ChangePipette/__tests__/ExitModal.test.tsx +++ b/app/src/organisms/ChangePipette/__tests__/ExitModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect, beforeEach } from 'vitest' diff --git a/app/src/organisms/ChangePipette/__tests__/InstructionStep.test.tsx b/app/src/organisms/ChangePipette/__tests__/InstructionStep.test.tsx index 33e390ad1f4..80cdde63972 100644 --- a/app/src/organisms/ChangePipette/__tests__/InstructionStep.test.tsx +++ b/app/src/organisms/ChangePipette/__tests__/InstructionStep.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { it, describe, beforeEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/ChangePipette/__tests__/Instructions.test.tsx b/app/src/organisms/ChangePipette/__tests__/Instructions.test.tsx index c169f5f0960..024be58e798 100644 --- a/app/src/organisms/ChangePipette/__tests__/Instructions.test.tsx +++ b/app/src/organisms/ChangePipette/__tests__/Instructions.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, describe, expect, beforeEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/organisms/ChangePipette/__tests__/LevelPipette.test.tsx b/app/src/organisms/ChangePipette/__tests__/LevelPipette.test.tsx index 40edcf61260..78c2e9f8d6d 100644 --- a/app/src/organisms/ChangePipette/__tests__/LevelPipette.test.tsx +++ b/app/src/organisms/ChangePipette/__tests__/LevelPipette.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, describe, expect, beforeEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/organisms/ChangePipette/__tests__/PipetteSelection.test.tsx b/app/src/organisms/ChangePipette/__tests__/PipetteSelection.test.tsx index 1048e265fae..f3619e9930d 100644 --- a/app/src/organisms/ChangePipette/__tests__/PipetteSelection.test.tsx +++ b/app/src/organisms/ChangePipette/__tests__/PipetteSelection.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, describe, beforeEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/ChangePipette/index.tsx b/app/src/organisms/ChangePipette/index.tsx index cd91b04529e..efcca7d7a35 100644 --- a/app/src/organisms/ChangePipette/index.tsx +++ b/app/src/organisms/ChangePipette/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useRef, useState, useEffect, useCallback } from 'react' import capitalize from 'lodash/capitalize' import { useSelector, useDispatch } from 'react-redux' import { useNavigate } from 'react-router-dom' @@ -31,7 +31,7 @@ import { import { WizardHeader } from '/app/molecules/WizardHeader' import { InProgressModal } from '/app/molecules/InProgressModal/InProgressModal' -import { useAttachedPipettes } from '../Devices/hooks' +import { useAttachedPipettes } from '/app/resources/instruments' import { ExitModal } from './ExitModal' import { Instructions } from './Instructions' import { ConfirmPipette } from './ConfirmPipette' @@ -63,7 +63,7 @@ export function ChangePipette(props: Props): JSX.Element | null { const { t } = useTranslation(['change_pipette', 'shared']) const navigate = useNavigate() const dispatch = useDispatch() - const finalRequestId = React.useRef(null) + const finalRequestId = useRef(null) const [dispatchApiRequests] = useDispatchApiRequests(dispatchedAction => { if ( dispatchedAction.type === HOME && @@ -74,10 +74,10 @@ export function ChangePipette(props: Props): JSX.Element | null { finalRequestId.current = dispatchedAction.meta.requestId } }) - const [wizardStep, setWizardStep] = React.useState(CLEAR_DECK) - const [wantedName, setWantedName] = React.useState(null) - const [confirmExit, setConfirmExit] = React.useState(false) - const [currentStepCount, setCurrentStepCount] = React.useState(0) + const [wizardStep, setWizardStep] = useState(CLEAR_DECK) + const [wantedName, setWantedName] = useState(null) + const [confirmExit, setConfirmExit] = useState(false) + const [currentStepCount, setCurrentStepCount] = useState(0) // @ts-expect-error(sa, 2021-05-27): avoiding src code change, use in operator to type narrow const wantedPipette = wantedName ? getPipetteNameSpecs(wantedName) : null const attachedPipette = useAttachedPipettes()[mount] @@ -90,10 +90,8 @@ export function ChangePipette(props: Props): JSX.Element | null { const [ wrongWantedPipette, setWrongWantedPipette, - ] = React.useState(wantedPipette) - const [confirmPipetteLevel, setConfirmPipetteLevel] = React.useState( - false - ) + ] = useState(wantedPipette) + const [confirmPipetteLevel, setConfirmPipetteLevel] = useState(false) const movementStatus = useSelector((state: State) => { return getMovementStatus(state, robotName) @@ -105,11 +103,11 @@ export function ChangePipette(props: Props): JSX.Element | null { : null })?.status - React.useEffect(() => { + useEffect(() => { if (homePipStatus === SUCCESS) closeModal() }, [homePipStatus, closeModal]) - const homePipAndExit = React.useCallback(() => { + const homePipAndExit = useCallback(() => { dispatchApiRequests(home(robotName, PIPETTE, mount)) }, [dispatchApiRequests, robotName, mount]) diff --git a/app/src/organisms/CheckCalibration/ResultsSummary/CalibrationHealthCheckResults.tsx b/app/src/organisms/CheckCalibration/ResultsSummary/CalibrationHealthCheckResults.tsx index f105a212ecc..957a91fb0eb 100644 --- a/app/src/organisms/CheckCalibration/ResultsSummary/CalibrationHealthCheckResults.tsx +++ b/app/src/organisms/CheckCalibration/ResultsSummary/CalibrationHealthCheckResults.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/CheckCalibration/ResultsSummary/CalibrationResult.tsx b/app/src/organisms/CheckCalibration/ResultsSummary/CalibrationResult.tsx index 9fd0252cc28..6d060e30b72 100644 --- a/app/src/organisms/CheckCalibration/ResultsSummary/CalibrationResult.tsx +++ b/app/src/organisms/CheckCalibration/ResultsSummary/CalibrationResult.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' diff --git a/app/src/organisms/CheckCalibration/ResultsSummary/RenderMountInformation.tsx b/app/src/organisms/CheckCalibration/ResultsSummary/RenderMountInformation.tsx index f93c62acdf6..fb79709cf33 100644 --- a/app/src/organisms/CheckCalibration/ResultsSummary/RenderMountInformation.tsx +++ b/app/src/organisms/CheckCalibration/ResultsSummary/RenderMountInformation.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/CheckCalibration/ResultsSummary/RenderResult.tsx b/app/src/organisms/CheckCalibration/ResultsSummary/RenderResult.tsx index 9f6fbe5e548..78c8f4e93d2 100644 --- a/app/src/organisms/CheckCalibration/ResultsSummary/RenderResult.tsx +++ b/app/src/organisms/CheckCalibration/ResultsSummary/RenderResult.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { ALIGN_CENTER, diff --git a/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/CalibrationHealthCheckResults.test.tsx b/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/CalibrationHealthCheckResults.test.tsx index 08e930a1391..dbe7c8349a0 100644 --- a/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/CalibrationHealthCheckResults.test.tsx +++ b/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/CalibrationHealthCheckResults.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { it, describe, expect, beforeEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/CalibrationResult.test.tsx b/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/CalibrationResult.test.tsx index 9004462430a..70432556944 100644 --- a/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/CalibrationResult.test.tsx +++ b/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/CalibrationResult.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, describe, beforeEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/RenderMountInformation.test.tsx b/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/RenderMountInformation.test.tsx index 1915e0ae711..bae133b6522 100644 --- a/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/RenderMountInformation.test.tsx +++ b/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/RenderMountInformation.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, describe, beforeEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/RenderResult.test.tsx b/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/RenderResult.test.tsx index 1fb708f81ba..90ed47fc126 100644 --- a/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/RenderResult.test.tsx +++ b/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/RenderResult.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { it, describe, expect, beforeEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/ResultsSummary.test.tsx b/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/ResultsSummary.test.tsx index 3f3ad46b7f1..5ec0334fdb5 100644 --- a/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/ResultsSummary.test.tsx +++ b/app/src/organisms/CheckCalibration/ResultsSummary/__tests__/ResultsSummary.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, it, describe, expect, beforeEach } from 'vitest' import { saveAs } from 'file-saver' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/organisms/CheckCalibration/ResultsSummary/index.tsx b/app/src/organisms/CheckCalibration/ResultsSummary/index.tsx index 06aec08bec3..e7d7e79c57e 100644 --- a/app/src/organisms/CheckCalibration/ResultsSummary/index.tsx +++ b/app/src/organisms/CheckCalibration/ResultsSummary/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { saveAs } from 'file-saver' diff --git a/app/src/organisms/CheckCalibration/ReturnTip.tsx b/app/src/organisms/CheckCalibration/ReturnTip.tsx index 02a9963be89..c90c8ec1a7a 100644 --- a/app/src/organisms/CheckCalibration/ReturnTip.tsx +++ b/app/src/organisms/CheckCalibration/ReturnTip.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { ALIGN_STRETCH, DIRECTION_COLUMN, diff --git a/app/src/organisms/CheckCalibration/ThresholdValue.tsx b/app/src/organisms/CheckCalibration/ThresholdValue.tsx index 4ff957fbb47..383f3f4ed31 100644 --- a/app/src/organisms/CheckCalibration/ThresholdValue.tsx +++ b/app/src/organisms/CheckCalibration/ThresholdValue.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - interface Props { thresholdVector: [number, number, number] } diff --git a/app/src/organisms/CheckCalibration/__tests__/CheckCalibration.test.tsx b/app/src/organisms/CheckCalibration/__tests__/CheckCalibration.test.tsx index 47ee094592f..d5f5a9814d7 100644 --- a/app/src/organisms/CheckCalibration/__tests__/CheckCalibration.test.tsx +++ b/app/src/organisms/CheckCalibration/__tests__/CheckCalibration.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { when } from 'vitest-when' import { vi, it, describe, expect, beforeEach, afterEach } from 'vitest' diff --git a/app/src/organisms/ChildNavigation/__tests__/ChildNavigation.test.tsx b/app/src/organisms/ChildNavigation/__tests__/ChildNavigation.test.tsx index 6bf5654398b..82a0dfb0b3c 100644 --- a/app/src/organisms/ChildNavigation/__tests__/ChildNavigation.test.tsx +++ b/app/src/organisms/ChildNavigation/__tests__/ChildNavigation.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect, beforeEach } from 'vitest' diff --git a/app/src/organisms/ChildNavigation/index.tsx b/app/src/organisms/ChildNavigation/index.tsx index 9c2b0596ab1..ea6c72f293b 100644 --- a/app/src/organisms/ChildNavigation/index.tsx +++ b/app/src/organisms/ChildNavigation/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import styled from 'styled-components' import { diff --git a/app/src/organisms/ChooseProtocolSlideout/__tests__/ChooseProtocolSlideout.test.tsx b/app/src/organisms/ChooseProtocolSlideout/__tests__/ChooseProtocolSlideout.test.tsx index 89ae868022a..a31c6a8236b 100644 --- a/app/src/organisms/ChooseProtocolSlideout/__tests__/ChooseProtocolSlideout.test.tsx +++ b/app/src/organisms/ChooseProtocolSlideout/__tests__/ChooseProtocolSlideout.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, describe, expect, beforeEach } from 'vitest' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen, waitFor } from '@testing-library/react' diff --git a/app/src/organisms/ChooseRobotSlideout/FileCard.tsx b/app/src/organisms/ChooseRobotSlideout/FileCard.tsx index 0ed23c4aa2f..e21e806c5cb 100644 --- a/app/src/organisms/ChooseRobotSlideout/FileCard.tsx +++ b/app/src/organisms/ChooseRobotSlideout/FileCard.tsx @@ -1,4 +1,3 @@ -import React from 'react' import { css } from 'styled-components' import { ALIGN_CENTER, diff --git a/app/src/organisms/ChooseRobotSlideout/__tests__/ChooseRobotSlideout.test.tsx b/app/src/organisms/ChooseRobotSlideout/__tests__/ChooseRobotSlideout.test.tsx index b9c3eab323c..998c82462bc 100644 --- a/app/src/organisms/ChooseRobotSlideout/__tests__/ChooseRobotSlideout.test.tsx +++ b/app/src/organisms/ChooseRobotSlideout/__tests__/ChooseRobotSlideout.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, describe, expect, beforeEach } from 'vitest' import { MemoryRouter } from 'react-router-dom' diff --git a/app/src/organisms/ChooseRobotSlideout/__tests__/FileCard.test.tsx b/app/src/organisms/ChooseRobotSlideout/__tests__/FileCard.test.tsx index eefe4f96241..9a151cd1704 100644 --- a/app/src/organisms/ChooseRobotSlideout/__tests__/FileCard.test.tsx +++ b/app/src/organisms/ChooseRobotSlideout/__tests__/FileCard.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, describe, expect } from 'vitest' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/organisms/ChooseRobotSlideout/index.tsx b/app/src/organisms/ChooseRobotSlideout/index.tsx index 109d1832163..1362e097a5b 100644 --- a/app/src/organisms/ChooseRobotSlideout/index.tsx +++ b/app/src/organisms/ChooseRobotSlideout/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useReducer, useEffect, Fragment } from 'react' import { useTranslation, Trans } from 'react-i18next' import { useSelector, useDispatch } from 'react-redux' import { NavLink } from 'react-router-dom' @@ -7,6 +7,7 @@ import { css } from 'styled-components' import { ALIGN_CENTER, ALIGN_FLEX_END, + Banner, BORDERS, COLORS, CURSOR_AUTO, @@ -47,7 +48,6 @@ import { RE_ROBOT_MODEL_OT2, RE_ROBOT_MODEL_OT3, } from '/app/redux/discovery' -import { Banner } from '/app/atoms/Banner' import { Slideout } from '/app/atoms/Slideout' import { MultiSlideout } from '/app/atoms/Slideout/MultiSlideout' import { ToggleButton } from '/app/atoms/buttons' @@ -162,8 +162,8 @@ export function ChooseRobotSlideout( const [ showRestoreValuesTooltip, setShowRestoreValuesTooltip, - ] = React.useState(false) - const [isInputFocused, setIsInputFocused] = React.useState(false) + ] = useState(false) + const [isInputFocused, setIsInputFocused] = useState(false) const unhealthyReachableRobots = useSelector((state: State) => getReachableRobots(state) @@ -199,7 +199,7 @@ export function ChooseRobotSlideout( } }) - const [robotBusyStatusByName, registerRobotBusyStatus] = React.useReducer( + const [robotBusyStatusByName, registerRobotBusyStatus] = useReducer( robotBusyStatusByNameReducer, {} ) @@ -212,7 +212,7 @@ export function ChooseRobotSlideout( ).length // this useEffect sets the default selection to the first robot in the list. state is managed by the caller - React.useEffect(() => { + useEffect(() => { if ( (selectedRobot == null || !reducerAvailableRobots.some( @@ -284,7 +284,7 @@ export function ChooseRobotSlideout( const isSelected = selectedRobot != null && selectedRobot.ip === robot.ip return ( - + { @@ -331,7 +331,7 @@ export function ChooseRobotSlideout( )} )} - + ) }) )} diff --git a/app/src/organisms/ChooseRobotToRunProtocolSlideout/__tests__/ChooseRobotToRunProtocolSlideout.test.tsx b/app/src/organisms/ChooseRobotToRunProtocolSlideout/__tests__/ChooseRobotToRunProtocolSlideout.test.tsx index 068cd0825f0..95e15dfcb77 100644 --- a/app/src/organisms/ChooseRobotToRunProtocolSlideout/__tests__/ChooseRobotToRunProtocolSlideout.test.tsx +++ b/app/src/organisms/ChooseRobotToRunProtocolSlideout/__tests__/ChooseRobotToRunProtocolSlideout.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, describe, expect, beforeEach, afterEach } from 'vitest' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen, waitFor } from '@testing-library/react' diff --git a/app/src/organisms/ConfigurePipette/ConfigErrorBanner.tsx b/app/src/organisms/ConfigurePipette/ConfigErrorBanner.tsx index 35aac004632..5a7ea72ae7e 100644 --- a/app/src/organisms/ConfigurePipette/ConfigErrorBanner.tsx +++ b/app/src/organisms/ConfigurePipette/ConfigErrorBanner.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { AlertItem } from '@opentrons/components' import styles from './styles.module.css' @@ -10,7 +10,7 @@ const TITLE = 'Error updating pipette settings' export function ConfigErrorBanner(props: Props): JSX.Element | null { const { message } = props - const [dismissed, setDismissed] = React.useState(false) + const [dismissed, setDismissed] = useState(false) if (message == null || dismissed) return null return ( diff --git a/app/src/organisms/ConfigurePipette/ConfigForm.tsx b/app/src/organisms/ConfigurePipette/ConfigForm.tsx index acc2c943cb9..6cd5691db1e 100644 --- a/app/src/organisms/ConfigurePipette/ConfigForm.tsx +++ b/app/src/organisms/ConfigurePipette/ConfigForm.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - import startCase from 'lodash/startCase' import mapValues from 'lodash/mapValues' import forOwn from 'lodash/forOwn' diff --git a/app/src/organisms/ConfigurePipette/ConfigFormGroup.tsx b/app/src/organisms/ConfigurePipette/ConfigFormGroup.tsx index a7f75904ae3..9c8a2c25acf 100644 --- a/app/src/organisms/ConfigurePipette/ConfigFormGroup.tsx +++ b/app/src/organisms/ConfigurePipette/ConfigFormGroup.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Controller } from 'react-hook-form' import { CheckboxField, diff --git a/app/src/organisms/ConfigurePipette/ConfigFormResetButton.tsx b/app/src/organisms/ConfigurePipette/ConfigFormResetButton.tsx index 8d640e89bda..3d31d8b7bab 100644 --- a/app/src/organisms/ConfigurePipette/ConfigFormResetButton.tsx +++ b/app/src/organisms/ConfigurePipette/ConfigFormResetButton.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { DIRECTION_COLUMN, diff --git a/app/src/organisms/ConfigurePipette/ConfigFormSubmitButton.tsx b/app/src/organisms/ConfigurePipette/ConfigFormSubmitButton.tsx index cc608f40e85..97335e736a3 100644 --- a/app/src/organisms/ConfigurePipette/ConfigFormSubmitButton.tsx +++ b/app/src/organisms/ConfigurePipette/ConfigFormSubmitButton.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { DIRECTION_COLUMN, diff --git a/app/src/organisms/ConfigurePipette/ConfigMessage.tsx b/app/src/organisms/ConfigurePipette/ConfigMessage.tsx index d55c2b96c36..d6a32fa6d4b 100644 --- a/app/src/organisms/ConfigurePipette/ConfigMessage.tsx +++ b/app/src/organisms/ConfigurePipette/ConfigMessage.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import styles from './styles.module.css' // TODO (ka 2019-2-12): Add intercom onClick to assistance text diff --git a/app/src/organisms/ConfigurePipette/__tests__/ConfigFormResetButton.test.tsx b/app/src/organisms/ConfigurePipette/__tests__/ConfigFormResetButton.test.tsx index bcec76bc2ea..4c2ec08c7d4 100644 --- a/app/src/organisms/ConfigurePipette/__tests__/ConfigFormResetButton.test.tsx +++ b/app/src/organisms/ConfigurePipette/__tests__/ConfigFormResetButton.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, expect, describe, beforeEach } from 'vitest' diff --git a/app/src/organisms/ConfigurePipette/__tests__/ConfigFormSubmitButton.test.tsx b/app/src/organisms/ConfigurePipette/__tests__/ConfigFormSubmitButton.test.tsx index 3a10eb988f0..3dbab681883 100644 --- a/app/src/organisms/ConfigurePipette/__tests__/ConfigFormSubmitButton.test.tsx +++ b/app/src/organisms/ConfigurePipette/__tests__/ConfigFormSubmitButton.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { it, expect, describe, beforeEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/ConfigurePipette/__tests__/ConfigurePipette.test.tsx b/app/src/organisms/ConfigurePipette/__tests__/ConfigurePipette.test.tsx index e3e37a76ff0..2d7790bdd24 100644 --- a/app/src/organisms/ConfigurePipette/__tests__/ConfigurePipette.test.tsx +++ b/app/src/organisms/ConfigurePipette/__tests__/ConfigurePipette.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { vi, it, expect, describe, beforeEach } from 'vitest' diff --git a/app/src/organisms/ConfigurePipette/index.tsx b/app/src/organisms/ConfigurePipette/index.tsx index 326f9e5792e..62a90721b26 100644 --- a/app/src/organisms/ConfigurePipette/index.tsx +++ b/app/src/organisms/ConfigurePipette/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { Box } from '@opentrons/components' diff --git a/app/src/organisms/Desktop/Labware/AddCustomLabwareSlideout/__tests__/AddCustomLabwareSlideout.test.tsx b/app/src/organisms/Desktop/Labware/AddCustomLabwareSlideout/__tests__/AddCustomLabwareSlideout.test.tsx index 451e63491fd..977f44f2cce 100644 --- a/app/src/organisms/Desktop/Labware/AddCustomLabwareSlideout/__tests__/AddCustomLabwareSlideout.test.tsx +++ b/app/src/organisms/Desktop/Labware/AddCustomLabwareSlideout/__tests__/AddCustomLabwareSlideout.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { describe, it, expect, vi, beforeEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/organisms/Desktop/Labware/AddCustomLabwareSlideout/index.tsx b/app/src/organisms/Desktop/Labware/AddCustomLabwareSlideout/index.tsx index b6bebd77fdd..1ded9827380 100644 --- a/app/src/organisms/Desktop/Labware/AddCustomLabwareSlideout/index.tsx +++ b/app/src/organisms/Desktop/Labware/AddCustomLabwareSlideout/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useDispatch } from 'react-redux' import { useTranslation, Trans } from 'react-i18next' import { diff --git a/app/src/organisms/Desktop/Labware/LabwareCard/__tests__/CustomLabwareOverflowMenu.test.tsx b/app/src/organisms/Desktop/Labware/LabwareCard/__tests__/CustomLabwareOverflowMenu.test.tsx index d96be28d824..588b47ecea3 100644 --- a/app/src/organisms/Desktop/Labware/LabwareCard/__tests__/CustomLabwareOverflowMenu.test.tsx +++ b/app/src/organisms/Desktop/Labware/LabwareCard/__tests__/CustomLabwareOverflowMenu.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, expect, beforeEach, afterEach } from 'vitest' diff --git a/app/src/organisms/Desktop/Labware/LabwareCard/__tests__/LabwareCard.test.tsx b/app/src/organisms/Desktop/Labware/LabwareCard/__tests__/LabwareCard.test.tsx index 35423948e56..76abb51c72f 100644 --- a/app/src/organisms/Desktop/Labware/LabwareCard/__tests__/LabwareCard.test.tsx +++ b/app/src/organisms/Desktop/Labware/LabwareCard/__tests__/LabwareCard.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, vi, beforeEach } from 'vitest' import { renderWithProviders, nestedTextMatcher } from '/app/__testing-utils__' diff --git a/app/src/organisms/Desktop/Labware/LabwareCard/index.tsx b/app/src/organisms/Desktop/Labware/LabwareCard/index.tsx index b94f088d87a..6d445b4bbd5 100644 --- a/app/src/organisms/Desktop/Labware/LabwareCard/index.tsx +++ b/app/src/organisms/Desktop/Labware/LabwareCard/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import startCase from 'lodash/startCase' import { format } from 'date-fns' diff --git a/app/src/organisms/Desktop/Labware/LabwareDetails/Dimensions.tsx b/app/src/organisms/Desktop/Labware/LabwareDetails/Dimensions.tsx index 61e3cd422d5..ad08946ea42 100644 --- a/app/src/organisms/Desktop/Labware/LabwareDetails/Dimensions.tsx +++ b/app/src/organisms/Desktop/Labware/LabwareDetails/Dimensions.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import round from 'lodash/round' import { Box, SPACING, getFootprintDiagram } from '@opentrons/components' diff --git a/app/src/organisms/Desktop/Labware/LabwareDetails/Gallery.tsx b/app/src/organisms/Desktop/Labware/LabwareDetails/Gallery.tsx index ca5f7b89de6..ed418218147 100644 --- a/app/src/organisms/Desktop/Labware/LabwareDetails/Gallery.tsx +++ b/app/src/organisms/Desktop/Labware/LabwareDetails/Gallery.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { Box, @@ -34,7 +34,7 @@ export function Gallery(props: GalleryProps): JSX.Element { ? 127.4 : dims.xDimension - const [currentImage, setCurrentImage] = React.useState(0) + const [currentImage, setCurrentImage] = useState(0) const render = ( 1 const isMultiRow = ordering.some(row => row.length > 1) const isCustomDefinition = definition.namespace !== 'opentrons' - const [showToolTip, setShowToolTip] = React.useState(false) + const [showToolTip, setShowToolTip] = useState(false) const [targetProps, tooltipProps] = useHoverTooltip({ placement: TOOLTIP_TOP_START, }) @@ -88,7 +88,7 @@ export function LabwareDetails(props: LabwareDetailsProps): JSX.Element { setShowToolTip(true) } - React.useEffect(() => { + useEffect(() => { const timer = setTimeout(() => { setShowToolTip(false) }, 2000) @@ -216,7 +216,7 @@ export function LabwareDetails(props: LabwareDetailsProps): JSX.Element { : '' return ( - + {groupMetadata.displayCategory == null && irregular && ( <> - + ) })} diff --git a/app/src/organisms/DeviceDetailsDeckConfiguration/AddFixtureModal.stories.tsx b/app/src/organisms/DeviceDetailsDeckConfiguration/AddFixtureModal.stories.tsx index de7c062f9d1..f14c6d46e96 100644 --- a/app/src/organisms/DeviceDetailsDeckConfiguration/AddFixtureModal.stories.tsx +++ b/app/src/organisms/DeviceDetailsDeckConfiguration/AddFixtureModal.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { QueryClient, QueryClientProvider } from 'react-query' import { VIEWPORT } from '@opentrons/components' import { AddFixtureModal } from './AddFixtureModal' diff --git a/app/src/organisms/DeviceDetailsDeckConfiguration/DeckConfigurationDiscardChangesModal.stories.tsx b/app/src/organisms/DeviceDetailsDeckConfiguration/DeckConfigurationDiscardChangesModal.stories.tsx index 0fdee52a94e..971950f222c 100644 --- a/app/src/organisms/DeviceDetailsDeckConfiguration/DeckConfigurationDiscardChangesModal.stories.tsx +++ b/app/src/organisms/DeviceDetailsDeckConfiguration/DeckConfigurationDiscardChangesModal.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { VIEWPORT } from '@opentrons/components' import { DeckConfigurationDiscardChangesModal } from './DeckConfigurationDiscardChangesModal' import type { Story, Meta } from '@storybook/react' diff --git a/app/src/organisms/DeviceDetailsDeckConfiguration/DeckConfigurationDiscardChangesModal.tsx b/app/src/organisms/DeviceDetailsDeckConfiguration/DeckConfigurationDiscardChangesModal.tsx index 9e803032b2d..bdf8339de71 100644 --- a/app/src/organisms/DeviceDetailsDeckConfiguration/DeckConfigurationDiscardChangesModal.tsx +++ b/app/src/organisms/DeviceDetailsDeckConfiguration/DeckConfigurationDiscardChangesModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useNavigate } from 'react-router-dom' diff --git a/app/src/organisms/DeviceDetailsDeckConfiguration/DeckFixtureSetupInstructionsModal.tsx b/app/src/organisms/DeviceDetailsDeckConfiguration/DeckFixtureSetupInstructionsModal.tsx index 8f7f72c3433..cfda966a3c1 100644 --- a/app/src/organisms/DeviceDetailsDeckConfiguration/DeckFixtureSetupInstructionsModal.tsx +++ b/app/src/organisms/DeviceDetailsDeckConfiguration/DeckFixtureSetupInstructionsModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { ALIGN_FLEX_END, diff --git a/app/src/organisms/DeviceDetailsDeckConfiguration/TouchScreenDeckFixtureSetupInstructionModal.stories.tsx b/app/src/organisms/DeviceDetailsDeckConfiguration/TouchScreenDeckFixtureSetupInstructionModal.stories.tsx index ec078d74eea..09586ea57f2 100644 --- a/app/src/organisms/DeviceDetailsDeckConfiguration/TouchScreenDeckFixtureSetupInstructionModal.stories.tsx +++ b/app/src/organisms/DeviceDetailsDeckConfiguration/TouchScreenDeckFixtureSetupInstructionModal.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { VIEWPORT } from '@opentrons/components' import { DeckFixtureSetupInstructionsModal } from './DeckFixtureSetupInstructionsModal' import type { Story, Meta } from '@storybook/react' diff --git a/app/src/organisms/DeviceDetailsDeckConfiguration/__tests__/AddFixtureModal.test.tsx b/app/src/organisms/DeviceDetailsDeckConfiguration/__tests__/AddFixtureModal.test.tsx index 3834d80f839..450a64cc0e6 100644 --- a/app/src/organisms/DeviceDetailsDeckConfiguration/__tests__/AddFixtureModal.test.tsx +++ b/app/src/organisms/DeviceDetailsDeckConfiguration/__tests__/AddFixtureModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach, vi, expect, afterEach } from 'vitest' diff --git a/app/src/organisms/DeviceDetailsDeckConfiguration/__tests__/DeckConfigurationDiscardChangesModal.test.tsx b/app/src/organisms/DeviceDetailsDeckConfiguration/__tests__/DeckConfigurationDiscardChangesModal.test.tsx index 538ee899f46..fd0e56ffa4d 100644 --- a/app/src/organisms/DeviceDetailsDeckConfiguration/__tests__/DeckConfigurationDiscardChangesModal.test.tsx +++ b/app/src/organisms/DeviceDetailsDeckConfiguration/__tests__/DeckConfigurationDiscardChangesModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach, vi, expect } from 'vitest' diff --git a/app/src/organisms/DeviceDetailsDeckConfiguration/__tests__/DeckFixtureSetupInstructionsModal.test.tsx b/app/src/organisms/DeviceDetailsDeckConfiguration/__tests__/DeckFixtureSetupInstructionsModal.test.tsx index 611f34b2600..ddc9ff33194 100644 --- a/app/src/organisms/DeviceDetailsDeckConfiguration/__tests__/DeckFixtureSetupInstructionsModal.test.tsx +++ b/app/src/organisms/DeviceDetailsDeckConfiguration/__tests__/DeckFixtureSetupInstructionsModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach, vi, expect } from 'vitest' diff --git a/app/src/organisms/DeviceDetailsDeckConfiguration/__tests__/DeviceDetailsDeckConfiguration.test.tsx b/app/src/organisms/DeviceDetailsDeckConfiguration/__tests__/DeviceDetailsDeckConfiguration.test.tsx index 731639882e8..dc47261fac8 100644 --- a/app/src/organisms/DeviceDetailsDeckConfiguration/__tests__/DeviceDetailsDeckConfiguration.test.tsx +++ b/app/src/organisms/DeviceDetailsDeckConfiguration/__tests__/DeviceDetailsDeckConfiguration.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { when } from 'vitest-when' import { describe, it, beforeEach, vi, afterEach } from 'vitest' diff --git a/app/src/organisms/DeviceDetailsDeckConfiguration/index.tsx b/app/src/organisms/DeviceDetailsDeckConfiguration/index.tsx index 2e717a4638a..da66506191c 100644 --- a/app/src/organisms/DeviceDetailsDeckConfiguration/index.tsx +++ b/app/src/organisms/DeviceDetailsDeckConfiguration/index.tsx @@ -1,10 +1,11 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' import { ALIGN_CENTER, ALIGN_FLEX_START, + Banner, BORDERS, COLORS, DeckConfigurator, @@ -29,7 +30,6 @@ import { } from '@opentrons/shared-data' import { useNotifyCurrentMaintenanceRun } from '/app/resources/maintenance_runs' -import { Banner } from '/app/atoms/Banner' import { DeckFixtureSetupInstructionsModal } from './DeckFixtureSetupInstructionsModal' import { useIsRobotViewable, useRunStatuses } from '../Devices/hooks' import { useIsEstopNotDisengaged } from '/app/resources/devices/hooks/useIsEstopNotDisengaged' @@ -58,7 +58,7 @@ export function DeviceDetailsDeckConfiguration({ const [ showSetupInstructionsModal, setShowSetupInstructionsModal, - ] = React.useState(false) + ] = useState(false) const { data: modulesData } = useModulesQuery() const deckConfig = diff --git a/app/src/organisms/Devices/CalibrationStatusBanner.tsx b/app/src/organisms/Devices/CalibrationStatusBanner.tsx index 0a16d516658..e601ea19799 100644 --- a/app/src/organisms/Devices/CalibrationStatusBanner.tsx +++ b/app/src/organisms/Devices/CalibrationStatusBanner.tsx @@ -1,9 +1,9 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { Link as RouterLink } from 'react-router-dom' import { ALIGN_CENTER, + Banner, COLORS, DIRECTION_ROW, Flex, @@ -14,7 +14,6 @@ import { TYPOGRAPHY, } from '@opentrons/components' -import { Banner } from '/app/atoms/Banner' import { useCalibrationTaskList } from './hooks' interface CalibrationStatusBannerProps { diff --git a/app/src/organisms/Devices/ConnectionTroubleshootingModal.tsx b/app/src/organisms/Devices/ConnectionTroubleshootingModal.tsx index c6289ead522..1587524abae 100644 --- a/app/src/organisms/Devices/ConnectionTroubleshootingModal.tsx +++ b/app/src/organisms/Devices/ConnectionTroubleshootingModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' import { diff --git a/app/src/organisms/Devices/DevicesEmptyState.tsx b/app/src/organisms/Devices/DevicesEmptyState.tsx index 95ec5f8af2c..f542fb2f979 100644 --- a/app/src/organisms/Devices/DevicesEmptyState.tsx +++ b/app/src/organisms/Devices/DevicesEmptyState.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useDispatch } from 'react-redux' diff --git a/app/src/organisms/Devices/DownloadCsvFileLink.tsx b/app/src/organisms/Devices/DownloadCsvFileLink.tsx index 4975db0ce11..e2ab564c0c8 100644 --- a/app/src/organisms/Devices/DownloadCsvFileLink.tsx +++ b/app/src/organisms/Devices/DownloadCsvFileLink.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/Devices/EstopBanner.tsx b/app/src/organisms/Devices/EstopBanner.tsx index f006be1ed47..3573bd67d59 100644 --- a/app/src/organisms/Devices/EstopBanner.tsx +++ b/app/src/organisms/Devices/EstopBanner.tsx @@ -1,7 +1,7 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { Btn, + Banner, DIRECTION_ROW, Flex, SPACING, @@ -9,7 +9,6 @@ import { TYPOGRAPHY, } from '@opentrons/components' -import { Banner } from '/app/atoms/Banner' import { NOT_PRESENT, PHYSICALLY_ENGAGED, diff --git a/app/src/organisms/Devices/HistoricalProtocolRun.tsx b/app/src/organisms/Devices/HistoricalProtocolRun.tsx index e843ecec15d..1ec488cfc45 100644 --- a/app/src/organisms/Devices/HistoricalProtocolRun.tsx +++ b/app/src/organisms/Devices/HistoricalProtocolRun.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' import { @@ -15,7 +15,7 @@ import { CURSOR_POINTER, } from '@opentrons/components' import { formatInterval } from '/app/transformations/commands' -import { formatTimestamp } from './utils' +import { formatTimestamp } from '/app/transformations/runs' import { EMPTY_TIMESTAMP } from '/app/resources/runs' import { HistoricalProtocolRunOverflowMenu as OverflowMenu } from './HistoricalProtocolRunOverflowMenu' import { HistoricalProtocolRunDrawer as Drawer } from './HistoricalProtocolRunDrawer' @@ -40,7 +40,7 @@ export function HistoricalProtocolRun( ): JSX.Element | null { const { t } = useTranslation('run_details') const { run, protocolName, robotIsBusy, robotName, protocolKey } = props - const [drawerOpen, setDrawerOpen] = React.useState(false) + const [drawerOpen, setDrawerOpen] = useState(false) const countRunDataFiles = 'runTimeParameters' in run ? run?.runTimeParameters.filter( diff --git a/app/src/organisms/Devices/HistoricalProtocolRunDrawer.tsx b/app/src/organisms/Devices/HistoricalProtocolRunDrawer.tsx index 26316a4b990..6533895bb1e 100644 --- a/app/src/organisms/Devices/HistoricalProtocolRunDrawer.tsx +++ b/app/src/organisms/Devices/HistoricalProtocolRunDrawer.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { format } from 'date-fns' import isEqual from 'lodash/isEqual' @@ -6,6 +5,7 @@ import { css } from 'styled-components' import { ALIGN_CENTER, ALIGN_END, + Banner, BORDERS, Box, COLORS, @@ -27,8 +27,7 @@ import { } from '@opentrons/shared-data' import { useCsvFileQuery } from '@opentrons/react-api-client' import { DownloadCsvFileLink } from './DownloadCsvFileLink' -import { Banner } from '/app/atoms/Banner' -import { useMostRecentCompletedAnalysis } from '../LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' import { useDeckCalibrationData } from './hooks' import { OffsetVector } from '/app/molecules/OffsetVector' import type { RunData } from '@opentrons/api-client' diff --git a/app/src/organisms/Devices/HistoricalProtocolRunOverflowMenu.tsx b/app/src/organisms/Devices/HistoricalProtocolRunOverflowMenu.tsx index 4e4723a5ccd..9fea6021603 100644 --- a/app/src/organisms/Devices/HistoricalProtocolRunOverflowMenu.tsx +++ b/app/src/organisms/Devices/HistoricalProtocolRunOverflowMenu.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { NavLink, useNavigate } from 'react-router-dom' @@ -134,7 +134,10 @@ function MenuDropdown(props: MenuDropdownProps): JSX.Element { } const trackEvent = useTrackEvent() const { trackProtocolRunEvent } = useTrackProtocolRunEvent(runId, robotName) - const { reset, isRunControlLoading } = useRunControls(runId, onResetSuccess) + const { reset, isResetRunLoading, isRunControlLoading } = useRunControls( + runId, + onResetSuccess + ) const { deleteRun } = useDeleteRunMutation() const robot = useRobot(robotName) const robotSerialNumber = @@ -190,7 +193,18 @@ function MenuDropdown(props: MenuDropdownProps): JSX.Element { } data-testid="RecentProtocolRun_OverflowMenu_rerunNow" > - {t('rerun_now')} + + {t('rerun_now')} + {isResetRunLoading ? ( + + ) : null} + {isRobotOnWrongVersionOfSoftware && ( diff --git a/app/src/organisms/Devices/InstrumentsAndModules.tsx b/app/src/organisms/Devices/InstrumentsAndModules.tsx index b8b2e786bb5..e5062d50881 100644 --- a/app/src/organisms/Devices/InstrumentsAndModules.tsx +++ b/app/src/organisms/Devices/InstrumentsAndModules.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { getPipetteModelSpecs, LEFT, RIGHT } from '@opentrons/shared-data' import { @@ -10,6 +9,7 @@ import { import { ALIGN_CENTER, ALIGN_FLEX_START, + Banner, COLORS, DIRECTION_COLUMN, Flex, @@ -20,13 +20,12 @@ import { TYPOGRAPHY, } from '@opentrons/components' -import { Banner } from '/app/atoms/Banner' import { PipetteRecalibrationWarning } from './PipetteCard/PipetteRecalibrationWarning' import { useCurrentRunId } from '/app/resources/runs' import { useIsFlex } from '/app/redux-resources/robots' import { ModuleCard } from '../ModuleCard' import { useIsRobotViewable, useRunStatuses } from './hooks' -import { getShowPipetteCalibrationWarning } from './utils' +import { getShowPipetteCalibrationWarning } from '/app/transformations/instruments' import { PipetteCard } from './PipetteCard' import { FlexPipetteCard } from './PipetteCard/FlexPipetteCard' import { GripperCard } from '../GripperCard' diff --git a/app/src/organisms/Devices/PipetteCard/AboutPipetteSlideout.tsx b/app/src/organisms/Devices/PipetteCard/AboutPipetteSlideout.tsx index 304d7be4b23..a8588a00778 100644 --- a/app/src/organisms/Devices/PipetteCard/AboutPipetteSlideout.tsx +++ b/app/src/organisms/Devices/PipetteCard/AboutPipetteSlideout.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { COLORS, diff --git a/app/src/organisms/Devices/PipetteCard/FlexPipetteCard.tsx b/app/src/organisms/Devices/PipetteCard/FlexPipetteCard.tsx index a1042e92881..be99be7c6af 100644 --- a/app/src/organisms/Devices/PipetteCard/FlexPipetteCard.tsx +++ b/app/src/organisms/Devices/PipetteCard/FlexPipetteCard.tsx @@ -3,6 +3,7 @@ import { Trans, useTranslation } from 'react-i18next' import { css } from 'styled-components' import { CURSOR_POINTER, + Banner, LegacyStyledText, SPACING, TYPOGRAPHY, @@ -17,7 +18,6 @@ import { useCurrentSubsystemUpdateQuery, useHost, } from '@opentrons/react-api-client' -import { Banner } from '/app/atoms/Banner' import { InstrumentCard } from '/app/molecules/InstrumentCard' import { ChoosePipette } from '../../PipetteWizardFlows/ChoosePipette' import { FLOWS } from '../../PipetteWizardFlows/constants' diff --git a/app/src/organisms/Devices/PipetteCard/PipetteOverflowMenu.tsx b/app/src/organisms/Devices/PipetteCard/PipetteOverflowMenu.tsx index da8656dae94..58fc6aca8f0 100644 --- a/app/src/organisms/Devices/PipetteCard/PipetteOverflowMenu.tsx +++ b/app/src/organisms/Devices/PipetteCard/PipetteOverflowMenu.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/Devices/PipetteCard/PipetteRecalibrationWarning.tsx b/app/src/organisms/Devices/PipetteCard/PipetteRecalibrationWarning.tsx index 452d97766d3..10fcee0f08c 100644 --- a/app/src/organisms/Devices/PipetteCard/PipetteRecalibrationWarning.tsx +++ b/app/src/organisms/Devices/PipetteCard/PipetteRecalibrationWarning.tsx @@ -1,18 +1,18 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { Box, + Banner, DIRECTION_COLUMN, Flex, SPACING, LegacyStyledText, TYPOGRAPHY, } from '@opentrons/components' -import { Banner } from '/app/atoms/Banner' export const PipetteRecalibrationWarning = (): JSX.Element | null => { const { t } = useTranslation('device_details') - const [showBanner, setShowBanner] = React.useState(true) + const [showBanner, setShowBanner] = useState(true) if (!showBanner) return null return ( diff --git a/app/src/organisms/Devices/PipetteCard/PipetteSettingsSlideout.tsx b/app/src/organisms/Devices/PipetteCard/PipetteSettingsSlideout.tsx index 9b6cdff72a7..6b004085284 100644 --- a/app/src/organisms/Devices/PipetteCard/PipetteSettingsSlideout.tsx +++ b/app/src/organisms/Devices/PipetteCard/PipetteSettingsSlideout.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { Flex } from '@opentrons/components' import { useUpdatePipetteSettingsMutation } from '@opentrons/react-api-client' diff --git a/app/src/organisms/Devices/PipetteCard/__tests__/AboutPipetteSlideout.test.tsx b/app/src/organisms/Devices/PipetteCard/__tests__/AboutPipetteSlideout.test.tsx index 84d9fef2a9f..dd2274a3ab3 100644 --- a/app/src/organisms/Devices/PipetteCard/__tests__/AboutPipetteSlideout.test.tsx +++ b/app/src/organisms/Devices/PipetteCard/__tests__/AboutPipetteSlideout.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, beforeEach, expect } from 'vitest' import '@testing-library/jest-dom/vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/Devices/PipetteCard/__tests__/FlexPipetteCard.test.tsx b/app/src/organisms/Devices/PipetteCard/__tests__/FlexPipetteCard.test.tsx index b10a0d2f446..f2b62b12efc 100644 --- a/app/src/organisms/Devices/PipetteCard/__tests__/FlexPipetteCard.test.tsx +++ b/app/src/organisms/Devices/PipetteCard/__tests__/FlexPipetteCard.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, afterEach, expect } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/Devices/PipetteCard/__tests__/PipetteCard.test.tsx b/app/src/organisms/Devices/PipetteCard/__tests__/PipetteCard.test.tsx index 91380e8d7aa..752a91eb502 100644 --- a/app/src/organisms/Devices/PipetteCard/__tests__/PipetteCard.test.tsx +++ b/app/src/organisms/Devices/PipetteCard/__tests__/PipetteCard.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/Devices/PipetteCard/__tests__/PipetteOverflowMenu.test.tsx b/app/src/organisms/Devices/PipetteCard/__tests__/PipetteOverflowMenu.test.tsx index 36eddad5674..155d955b6ea 100644 --- a/app/src/organisms/Devices/PipetteCard/__tests__/PipetteOverflowMenu.test.tsx +++ b/app/src/organisms/Devices/PipetteCard/__tests__/PipetteOverflowMenu.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/Devices/PipetteCard/__tests__/PipetteSettingsSlideout.test.tsx b/app/src/organisms/Devices/PipetteCard/__tests__/PipetteSettingsSlideout.test.tsx index a1b0cdaa7ec..37b6f66b863 100644 --- a/app/src/organisms/Devices/PipetteCard/__tests__/PipetteSettingsSlideout.test.tsx +++ b/app/src/organisms/Devices/PipetteCard/__tests__/PipetteSettingsSlideout.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { fireEvent, waitFor, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/Devices/PipetteCard/index.tsx b/app/src/organisms/Devices/PipetteCard/index.tsx index 3e031da68e7..fa2da081457 100644 --- a/app/src/organisms/Devices/PipetteCard/index.tsx +++ b/app/src/organisms/Devices/PipetteCard/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { @@ -68,9 +68,9 @@ export const PipetteCard = (props: PipetteCardProps): JSX.Element => { setShowOverflowMenu(false) }, }) - const [showChangePipette, setChangePipette] = React.useState(false) - const [showSlideout, setShowSlideout] = React.useState(false) - const [showAboutSlideout, setShowAboutSlideout] = React.useState(false) + const [showChangePipette, setChangePipette] = useState(false) + const [showSlideout, setShowSlideout] = useState(false) + const [showAboutSlideout, setShowAboutSlideout] = useState(false) const { showDTWiz, toggleDTWiz } = useDropTipWizardFlows() diff --git a/app/src/organisms/Devices/ProtocolRun/BackToTopButton.tsx b/app/src/organisms/Devices/ProtocolRun/BackToTopButton.tsx index 385d6537272..a8524988bf2 100644 --- a/app/src/organisms/Devices/ProtocolRun/BackToTopButton.tsx +++ b/app/src/organisms/Devices/ProtocolRun/BackToTopButton.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { Link } from 'react-router-dom' import { useRobot } from '/app/redux-resources/robots' diff --git a/app/src/organisms/Devices/ProtocolRun/EmptySetupStep.tsx b/app/src/organisms/Devices/ProtocolRun/EmptySetupStep.tsx index eeca6dcf81a..24c2b449083 100644 --- a/app/src/organisms/Devices/ProtocolRun/EmptySetupStep.tsx +++ b/app/src/organisms/Devices/ProtocolRun/EmptySetupStep.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { COLORS, DIRECTION_COLUMN, diff --git a/app/src/organisms/Devices/ProtocolRun/LabwareInfoOverlay.tsx b/app/src/organisms/Devices/ProtocolRun/LabwareInfoOverlay.tsx index 1dc1b9526cb..c2d868af33f 100644 --- a/app/src/organisms/Devices/ProtocolRun/LabwareInfoOverlay.tsx +++ b/app/src/organisms/Devices/ProtocolRun/LabwareInfoOverlay.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { useTranslation } from 'react-i18next' import { getLabwareDisplayName } from '@opentrons/shared-data' diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/DisplayRunStatus.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/DisplayRunStatus.tsx index 70b388ec3ca..5b5624ffb4e 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/DisplayRunStatus.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/DisplayRunStatus.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderBannerContainer/ProtocolAnalysisErrorBanner.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderBannerContainer/ProtocolAnalysisErrorBanner.tsx index 183e51fb1eb..1f4501d74b6 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderBannerContainer/ProtocolAnalysisErrorBanner.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderBannerContainer/ProtocolAnalysisErrorBanner.tsx @@ -4,6 +4,7 @@ import { Trans, useTranslation } from 'react-i18next' import { ALIGN_CENTER, + Banner, Btn, Flex, JUSTIFY_FLEX_END, @@ -16,7 +17,6 @@ import { } from '@opentrons/components' import { getTopPortalEl } from '../../../../../App/portal' -import { Banner } from '/app/atoms/Banner' import type { AnalysisError } from '@opentrons/shared-data' diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderBannerContainer/TerminalRunBannerContainer.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderBannerContainer/TerminalRunBannerContainer.tsx index 3df355a0b3c..c6428c2f385 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderBannerContainer/TerminalRunBannerContainer.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderBannerContainer/TerminalRunBannerContainer.tsx @@ -1,9 +1,9 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { SPACING, TYPOGRAPHY, + Banner, JUSTIFY_SPACE_BETWEEN, Flex, StyledText, @@ -11,8 +11,6 @@ import { ALIGN_CENTER, } from '@opentrons/components' import { RUN_STATUS_STOPPED, RUN_STATUS_SUCCEEDED } from '@opentrons/api-client' - -import { Banner } from '/app/atoms/Banner' import { useCloseCurrentRun, useIsRunCurrent, diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderBannerContainer/__tests__/ProtocolAnalysisErrorBanner.test.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderBannerContainer/__tests__/ProtocolAnalysisErrorBanner.test.tsx index 169dee8958a..5b60de044d7 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderBannerContainer/__tests__/ProtocolAnalysisErrorBanner.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderBannerContainer/__tests__/ProtocolAnalysisErrorBanner.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach } from 'vitest' diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderBannerContainer/index.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderBannerContainer/index.tsx index d2e41230640..5a14054d98d 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderBannerContainer/index.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderBannerContainer/index.tsx @@ -1,10 +1,8 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' -import { Box, SPACING } from '@opentrons/components' +import { Box, SPACING, Banner } from '@opentrons/components' import { ProtocolAnalysisErrorBanner } from './ProtocolAnalysisErrorBanner' -import { Banner } from '/app/atoms/Banner' import { TerminalRunBannerContainer, useTerminalRunBannerContainer, diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/ActionButton/hooks/useActionBtnDisabledUtils.ts b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/ActionButton/hooks/useActionBtnDisabledUtils.ts index 1af46b835d6..df418deda15 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/ActionButton/hooks/useActionBtnDisabledUtils.ts +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/ActionButton/hooks/useActionBtnDisabledUtils.ts @@ -19,6 +19,7 @@ interface UseActionButtonDisabledUtilsProps extends BaseActionButtonProps { isOtherRunCurrent: boolean isProtocolNotReady: boolean isRobotOnWrongVersionOfSoftware: boolean + isClosingCurrentRun: boolean } type UseActionButtonDisabledUtilsResult = @@ -41,6 +42,7 @@ export function useActionBtnDisabledUtils( robotName, runId, isResetRunLoadingRef, + isClosingCurrentRun, } = props const { t } = useTranslation('shared') @@ -57,6 +59,7 @@ export function useActionBtnDisabledUtils( isPlayRunActionLoading || isPauseRunActionLoading || isResetRunLoading || + isClosingCurrentRun || isOtherRunCurrent || isProtocolNotReady || isFixtureMismatch || @@ -82,6 +85,7 @@ type UseDisabledReasonProps = UseActionButtonDisabledUtilsProps & { isDoorOpen: boolean isFixtureMismatch: boolean isResetRunLoading: boolean + isClosingCurrentRun: boolean } // The user-facing disabled explanation for why the ActionButton is disabled, if any. @@ -95,6 +99,7 @@ function useDisabledReason({ isDoorOpen, runStatus, isResetRunLoading, + isClosingCurrentRun, }: UseDisabledReasonProps): string | null { const { t } = useTranslation(['run_details', 'shared']) @@ -110,6 +115,8 @@ function useDisabledReason({ return t('shared:a_software_update_is_available') } else if (isDoorOpen && isStartRunStatus(runStatus)) { return t('close_door') + } else if (isClosingCurrentRun) { + return t('shared:robot_is_busy') } else { return null } diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/ActionButton/hooks/useActionButtonProperties.ts b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/ActionButton/hooks/useActionButtonProperties.ts index d47788130c4..5d37c134470 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/ActionButton/hooks/useActionButtonProperties.ts +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/ActionButton/hooks/useActionButtonProperties.ts @@ -35,6 +35,7 @@ interface UseButtonPropertiesProps extends BaseActionButtonProps { isValidRunAgain: boolean isOtherRunCurrent: boolean isRobotOnWrongVersionOfSoftware: boolean + isClosingCurrentRun: boolean } // Returns ActionButton properties. @@ -52,6 +53,7 @@ export function useActionButtonProperties({ attachedModules, runHeaderModalContainerUtils, isResetRunLoadingRef, + isClosingCurrentRun, }: UseButtonPropertiesProps): { buttonText: string handleButtonClick: () => void @@ -72,6 +74,9 @@ export function useActionButtonProperties({ if (isProtocolNotReady) { buttonIconName = 'ot-spinner' buttonText = t('analyzing_on_robot') + } else if (isClosingCurrentRun) { + buttonIconName = 'ot-spinner' + buttonText = t('canceling_run') } else if (runStatus === RUN_STATUS_RUNNING || isRecoveryStatus(runStatus)) { buttonIconName = 'pause' buttonText = t('pause_run') diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/ActionButton/hooks/useIsFixtureMismatch.ts b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/ActionButton/hooks/useIsFixtureMismatch.ts index ebac2a7bec3..aa49528efa2 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/ActionButton/hooks/useIsFixtureMismatch.ts +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/ActionButton/hooks/useIsFixtureMismatch.ts @@ -1,4 +1,4 @@ -import { useMostRecentCompletedAnalysis } from '../../../../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' import { useRobotType } from '/app/redux-resources/robots' import { getIsFixtureMismatch, diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/ActionButton/index.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/ActionButton/index.tsx index 00bca7d21b7..4b8c0f68076 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/ActionButton/index.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/ActionButton/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { RUN_STATUS_STOP_REQUESTED } from '@opentrons/api-client' import { @@ -14,14 +14,16 @@ import { useHoverTooltip, } from '@opentrons/components' +import { useRobot } from '/app/redux-resources/robots' +import { useRobotAnalyticsData } from '/app/redux-resources/analytics' import { - useModuleCalibrationStatus, + useCloseCurrentRun, + useCurrentRunId, + useProtocolDetailsForRun, useRunCalibrationStatus, useUnmatchedModulesForProtocol, -} from '../../../../hooks' -import { useRobot } from '/app/redux-resources/robots' -import { useRobotAnalyticsData } from '/app/redux-resources/analytics' -import { useCurrentRunId, useProtocolDetailsForRun } from '/app/resources/runs' + useModuleCalibrationStatus, +} from '/app/resources/runs' import { useActionBtnDisabledUtils, useActionButtonProperties } from './hooks' import { getFallbackRobotSerialNumber, isRunAgainStatus } from '../../utils' import { useIsRobotOnWrongVersionOfSoftware } from '/app/redux/robot-update' @@ -71,6 +73,7 @@ export function ActionButton(props: ActionButtonProps): JSX.Element { const isOtherRunCurrent = currentRunId != null && currentRunId !== runId const isProtocolNotReady = protocolData == null || !!isProtocolAnalyzing const isValidRunAgain = isRunAgainStatus(runStatus) + const { isClosingCurrentRun } = useCloseCurrentRun() const { isDisabled, disabledReason } = useActionBtnDisabledUtils({ isCurrentRun, @@ -79,6 +82,7 @@ export function ActionButton(props: ActionButtonProps): JSX.Element { isProtocolNotReady, isRobotOnWrongVersionOfSoftware, isValidRunAgain, + isClosingCurrentRun, ...props, }) @@ -102,6 +106,7 @@ export function ActionButton(props: ActionButtonProps): JSX.Element { isValidRunAgain, isOtherRunCurrent, isRobotOnWrongVersionOfSoftware, + isClosingCurrentRun, ...props, }) @@ -126,7 +131,8 @@ export function ActionButton(props: ActionButtonProps): JSX.Element { spin={ isProtocolNotReady || runStatus === RUN_STATUS_STOP_REQUESTED || - isResetRunLoadingRef.current + isResetRunLoadingRef.current || + isClosingCurrentRun } /> ) : null} diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/LabeledValue.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/LabeledValue.tsx index 0d839adbc93..135dd72bbae 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/LabeledValue.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/LabeledValue.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { DIRECTION_COLUMN, diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/RunHeaderSectionLower.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/RunHeaderSectionLower.tsx index 9c8168c7f11..d7757cd4fe9 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/RunHeaderSectionLower.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/RunHeaderSectionLower.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' @@ -14,8 +13,8 @@ import { } from '@opentrons/components' import { RUN_STATUS_RUNNING } from '@opentrons/api-client' -import { formatTimestamp } from '../../../utils' -import { useRunControls } from '../../../../RunTimeControl/hooks' +import { formatTimestamp } from '/app/transformations/runs' +import { useRunControls } from '/app/organisms/RunTimeControl/hooks' import { EMPTY_TIMESTAMP, useRunTimestamps, diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/RunHeaderSectionUpper.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/RunHeaderSectionUpper.tsx index df397d937d1..ecbee666819 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/RunHeaderSectionUpper.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/RunHeaderSectionUpper.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' @@ -11,10 +10,9 @@ import { import { LabeledValue } from './LabeledValue' import { DisplayRunStatus } from '../DisplayRunStatus' -import { RunTimer } from '../../RunTimer' +import { RunTimer } from '/app/molecules/RunTimer' import { ActionButton } from './ActionButton' -import { useRunCreatedAtTimestamp } from '../../../hooks' -import { useRunTimestamps } from '/app/resources/runs' +import { useRunTimestamps, useRunCreatedAtTimestamp } from '/app/resources/runs' import type { RunHeaderContentProps } from '.' diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/index.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/index.tsx index bb5f1779ea3..b62c0701563 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/index.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderContent/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { RunHeaderSectionUpper } from './RunHeaderSectionUpper' import { RunHeaderSectionLower } from './RunHeaderSectionLower' diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/RunHeaderModalContainer.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/RunHeaderModalContainer.tsx index 57d85a7e5c4..83be6db31ee 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/RunHeaderModalContainer.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/RunHeaderModalContainer.tsx @@ -1,8 +1,6 @@ -import * as React from 'react' - import { ErrorRecoveryFlows } from '../../../../ErrorRecoveryFlows' import { DropTipWizardFlows } from '../../../../DropTipWizardFlows' -import { useMostRecentCompletedAnalysis } from '../../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' import { ConfirmCancelModal, HeaterShakerIsRunningModal, diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/hooks/useRunHeaderDropTip.ts b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/hooks/useRunHeaderDropTip.ts index 77f9c35ce8f..4138dca538b 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/hooks/useRunHeaderDropTip.ts +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/hooks/useRunHeaderDropTip.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useEffect } from 'react' import { useHost } from '@opentrons/react-api-client' import { RUN_STATUS_IDLE, RUN_STATUS_STOPPED } from '@opentrons/api-client' @@ -103,7 +103,7 @@ export function useRunHeaderDropTip({ } // Manage tip checking - React.useEffect(() => { + useEffect(() => { // If a user begins a new run without navigating away from the run page, reset tip status. if (robotType === FLEX_ROBOT_TYPE) { if (runStatus === RUN_STATUS_IDLE) { @@ -122,7 +122,7 @@ export function useRunHeaderDropTip({ // If the run terminates with a "stopped" status, close the run if no tips are attached after running tip check at least once. // This marks the robot as "not busy" if drop tip CTAs are unnecessary. - React.useEffect(() => { + useEffect(() => { if ( runStatus === RUN_STATUS_STOPPED && isRunCurrent && diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/ConfirmMissingStepsModal.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/ConfirmMissingStepsModal.tsx index 708f07b3789..978efdbab48 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/ConfirmMissingStepsModal.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/ConfirmMissingStepsModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { ALIGN_CENTER, diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/HeaterShakerIsRunningModal.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/HeaterShakerIsRunningModal.tsx index ff69b0f9102..1db32eb66b2 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/HeaterShakerIsRunningModal.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/HeaterShakerIsRunningModal.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { useCreateLiveCommandMutation } from '@opentrons/react-api-client' @@ -18,7 +18,7 @@ import { } from '@opentrons/components' import { HEATERSHAKER_MODULE_TYPE } from '@opentrons/shared-data' -import { useAttachedModules } from '../../../../../hooks' +import { useAttachedModules } from '/app/resources/modules' import { HeaterShakerModuleCard } from './HeaterShakerModuleCard' import { getActiveHeaterShaker } from './utils' import { useIsHeaterShakerInProtocol } from '../../../../../../ModuleCard/hooks' @@ -34,7 +34,7 @@ export type UseHeaterShakerIsRunningModalResult = export function useHeaterShakerIsRunningModal( attachedModules: AttachedModule[] ): UseHeaterShakerIsRunningModalResult { - const [showIsShakingModal, setShowIsShakingModal] = React.useState(false) + const [showIsShakingModal, setShowIsShakingModal] = useState(false) const activeHeaterShaker = getActiveHeaterShaker(attachedModules) const isHeaterShakerInProtocol = useIsHeaterShakerInProtocol() diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/HeaterShakerModuleCard.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/HeaterShakerModuleCard.tsx index 8e2305926c6..ab12b127489 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/HeaterShakerModuleCard.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/HeaterShakerModuleCard.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/__tests__/HeaterShakerIsRunningModal.test.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/__tests__/HeaterShakerIsRunningModal.test.tsx index bcda9d52c3a..03b59af1b57 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/__tests__/HeaterShakerIsRunningModal.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/__tests__/HeaterShakerIsRunningModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' @@ -9,8 +9,8 @@ import { renderWithProviders } from '/app/__testing-utils__' import { mockHeaterShaker } from '/app/redux/modules/__fixtures__' import { HeaterShakerIsRunningModal } from '../HeaterShakerIsRunningModal' import { HeaterShakerModuleCard } from '../HeaterShakerModuleCard' -import { useAttachedModules } from '../../../../../../hooks' -import { useMostRecentCompletedAnalysis } from '../../../../../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { useAttachedModules } from '/app/resources/modules' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' import type * as ReactApiClient from '@opentrons/react-api-client' @@ -21,10 +21,8 @@ vi.mock('@opentrons/react-api-client', async importOriginal => { useCreateLiveCommandMutation: vi.fn(), } }) -vi.mock('../../../../../../hooks') -vi.mock( - '../../../../../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' -) +vi.mock('/app/resources/modules') +vi.mock('/app/resources/runs') vi.mock('../HeaterShakerModuleCard') const mockMovingHeaterShakerOne = { diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/__tests__/HeaterShakerModuleCard.test.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/__tests__/HeaterShakerModuleCard.test.tsx index b3ca63fb0c8..82b12033164 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/__tests__/HeaterShakerModuleCard.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/__tests__/HeaterShakerModuleCard.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/__tests__/hooks.test.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/__tests__/hooks.test.tsx index 75941043fd3..c5028c6a821 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/__tests__/hooks.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/__tests__/hooks.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Provider } from 'react-redux' import { describe, it, vi, beforeEach, expect } from 'vitest' import { createStore } from 'redux' @@ -7,15 +7,13 @@ import { renderHook } from '@testing-library/react' import { HEATERSHAKER_MODULE_V1 } from '@opentrons/shared-data' import { RUN_ID_1 } from '/app/resources/runs/__fixtures__' -import { useMostRecentCompletedAnalysis } from '../../../../../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' import { useHeaterShakerModuleIdsFromRun } from '../hooks' import type { Store } from 'redux' import type { State } from '/app/redux/types' -vi.mock( - '../../../../../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' -) +vi.mock('/app/resources/runs') describe('useHeaterShakerModuleIdsFromRun', () => { const store: Store = createStore(vi.fn(), {}) diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/hooks.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/hooks.tsx index 680077a1aa6..903327c8834 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/hooks.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/HeaterShakerIsRunningModal/hooks.tsx @@ -1,4 +1,4 @@ -import { useMostRecentCompletedAnalysis } from '../../../../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' export interface ModuleIdsFromRun { moduleIdsFromRun: string[] diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/ProtocolAnalysisErrorModal.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/ProtocolAnalysisErrorModal.tsx index fbf911f229d..5a19c099a34 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/ProtocolAnalysisErrorModal.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/ProtocolAnalysisErrorModal.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' @@ -13,8 +13,8 @@ import { TYPOGRAPHY, } from '@opentrons/components' -import { getTopPortalEl } from '../../../../../../App/portal' -import { useProtocolAnalysisErrors } from '../../../../hooks' +import { getTopPortalEl } from '/app/App/portal' +import { useProtocolAnalysisErrors } from '/app/resources/runs' import type { AnalysisError } from '@opentrons/shared-data' @@ -34,9 +34,9 @@ export function useProtocolAnalysisErrorsModal({ runId, }: UseAnalysisErrorsModalProps): UseAnalysisErrorsModalResult { const { analysisErrors } = useProtocolAnalysisErrors(runId) - const [showModal, setShowModal] = React.useState(false) + const [showModal, setShowModal] = useState(false) - React.useEffect(() => { + useEffect(() => { if (analysisErrors != null && analysisErrors?.length > 0) { setShowModal(true) } diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/ProtocolDropTipModal.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/ProtocolDropTipModal.tsx index b4bad8d9c8d..b475ea725e6 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/ProtocolDropTipModal.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/ProtocolDropTipModal.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { Trans, useTranslation } from 'react-i18next' import { css } from 'styled-components' @@ -52,7 +52,7 @@ export function useProtocolDropTipModal({ onSkipAndHome, pipetteInfo, }: UseProtocolDropTipModalProps): UseProtocolDropTipModalResult { - const [showModal, setShowModal] = React.useState(areTipsAttached) + const [showModal, setShowModal] = useState(areTipsAttached) const { homePipettes, isHoming } = useHomePipettes({ pipetteInfo, @@ -62,7 +62,7 @@ export function useProtocolDropTipModal({ }) // Close the modal if a different app closes the run context. - React.useEffect(() => { + useEffect(() => { if (isRunCurrent && !isHoming) { setShowModal(areTipsAttached) } else if (!isRunCurrent) { diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/__tests__/ConfirmCancelModal.test.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/__tests__/ConfirmCancelModal.test.tsx index c12f6a150cf..c6421040e17 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/__tests__/ConfirmCancelModal.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/__tests__/ConfirmCancelModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest' diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/__tests__/ProtocolAnalysisErrorModal.test.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/__tests__/ProtocolAnalysisErrorModal.test.tsx index 06d462abd1a..44fcb0278ad 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/__tests__/ProtocolAnalysisErrorModal.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/__tests__/ProtocolAnalysisErrorModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach, expect, vi } from 'vitest' diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/__tests__/ProtocolDropTipModal.test.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/__tests__/ProtocolDropTipModal.test.tsx index 613b888a27b..1d73cc5c1cf 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/__tests__/ProtocolDropTipModal.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/__tests__/ProtocolDropTipModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, expect, beforeEach } from 'vitest' import { renderHook, act, screen, fireEvent } from '@testing-library/react' diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/__tests__/RunFailedModal.test.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/__tests__/RunFailedModal.test.tsx index 8664ce55b85..d49875a0859 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/__tests__/RunFailedModal.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderModalContainer/modals/__tests__/RunFailedModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, beforeEach, vi, expect, afterEach } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderProtocolName.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderProtocolName.tsx index 9e3addadbc2..f08a157bba9 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderProtocolName.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/RunHeaderProtocolName.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Link } from 'react-router-dom' import { diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/__tests__/ProtocolRunHeader.test.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/__tests__/ProtocolRunHeader.test.tsx index e06718390c6..f9996372b8a 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/__tests__/ProtocolRunHeader.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/__tests__/ProtocolRunHeader.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, expect, beforeEach, afterEach } from 'vitest' import { screen } from '@testing-library/react' import { useNavigate } from 'react-router-dom' diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/hooks/useRunAnalytics.ts b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/hooks/useRunAnalytics.ts index 896f432336c..31399cbc541 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/hooks/useRunAnalytics.ts +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader/hooks/useRunAnalytics.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useEffect } from 'react' import { useRobotAnalyticsData, useTrackProtocolRunEvent, @@ -26,7 +26,7 @@ export function useRunAnalytics({ const runStatus = useRunStatus(runId) const isRunCurrent = useIsRunCurrent(runId) - React.useEffect(() => { + useEffect(() => { const areReportConditionsValid = isRunCurrent && runId != null && @@ -42,7 +42,7 @@ export function useRunAnalytics({ }, [runStatus, isRunCurrent, runId, robotAnalyticsData]) const { reportRecoveredRunResult } = useRecoveryAnalytics() - React.useEffect(() => { + useEffect(() => { if (isRunCurrent) { reportRecoveredRunResult(runStatus, enteredER) } diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunModuleControls.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunModuleControls.tsx index 1ed25b251c4..c92251cfea1 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunModuleControls.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunModuleControls.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useInstrumentsQuery } from '@opentrons/react-api-client' import { useTranslation } from 'react-i18next' import { @@ -9,9 +8,9 @@ import { InfoScreen, SPACING, } from '@opentrons/components' -import { ModuleCard } from '../../ModuleCard' -import { useModuleRenderInfoForProtocolById } from '../hooks' -import { useModuleApiRequests } from '../../ModuleCard/utils' +import { ModuleCard } from '/app/organisms/ModuleCard' +import { useModuleRenderInfoForProtocolById } from '/app/resources/runs' +import { useModuleApiRequests } from '/app/organisms/ModuleCard/utils' import type { BadPipette, PipetteData } from '@opentrons/api-client' diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunRunTimeParameters.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunRunTimeParameters.tsx index f5fc97c22e2..a284b044283 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunRunTimeParameters.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunRunTimeParameters.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import styled, { css } from 'styled-components' @@ -13,6 +12,7 @@ import { } from '@opentrons/shared-data' import { ALIGN_CENTER, + Banner, BORDERS, Chip, COLORS, @@ -31,10 +31,12 @@ import { useHoverTooltip, } from '@opentrons/components' -import { Banner } from '/app/atoms/Banner' import { Divider } from '/app/atoms/structure' -import { useMostRecentCompletedAnalysis } from '../../LabwarePositionCheck/useMostRecentCompletedAnalysis' -import { useNotifyRunQuery, useRunStatus } from '/app/resources/runs' +import { + useMostRecentCompletedAnalysis, + useNotifyRunQuery, + useRunStatus, +} from '/app/resources/runs' import type { RunTimeParameter } from '@opentrons/shared-data' import type { RunStatus } from '@opentrons/api-client' diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunSetup.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunSetup.tsx index e216298f07a..0f48d0bb833 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunSetup.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunSetup.tsx @@ -32,14 +32,14 @@ import { useDeckConfigurationCompatibility } from '/app/resources/deck_configura import { useRobot, useIsFlex } from '/app/redux-resources/robots' import { useStoredProtocolAnalysis } from '/app/resources/analysis' import { - useModuleCalibrationStatus, - useProtocolAnalysisErrors, + useMostRecentCompletedAnalysis, + useRunPipetteInfoByMount, useRunCalibrationStatus, useRunHasStarted, - useRunPipetteInfoByMount, useUnmatchedModulesForProtocol, -} from '../hooks' -import { useMostRecentCompletedAnalysis } from '../../LabwarePositionCheck/useMostRecentCompletedAnalysis' + useModuleCalibrationStatus, + useProtocolAnalysisErrors, +} from '/app/resources/runs' import { SetupLabware } from './SetupLabware' import { SetupLabwarePositionCheck } from './SetupLabwarePositionCheck' import { SetupRobotCalibration } from './SetupRobotCalibration' @@ -184,8 +184,13 @@ export function ProtocolRunSetup({ setLabwareSetupComplete, ] = React.useState(false) const [liquidSetupComplete, setLiquidSetupComplete] = React.useState( - !hasLiquids + false ) + React.useEffect(() => { + if ((robotProtocolAnalysis || storedProtocolAnalysis) && !hasLiquids) { + setLiquidSetupComplete(true) + } + }, [robotProtocolAnalysis, storedProtocolAnalysis, hasLiquids]) if ( !hasLiquids && protocolAnalysis != null && diff --git a/app/src/organisms/Devices/ProtocolRun/SetupCalibrationItem.tsx b/app/src/organisms/Devices/ProtocolRun/SetupCalibrationItem.tsx index 89bec05607c..8ac68de4ff1 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupCalibrationItem.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupCalibrationItem.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { @@ -16,8 +15,8 @@ import { TYPOGRAPHY, } from '@opentrons/components' -import { useRunHasStarted } from '../hooks' -import { formatTimestamp } from '../utils' +import { useRunHasStarted } from '/app/resources/runs' +import { formatTimestamp } from '/app/transformations/runs' interface SetupCalibrationItemProps { calibratedDate: string | null diff --git a/app/src/organisms/Devices/ProtocolRun/SetupDeckCalibration.tsx b/app/src/organisms/Devices/ProtocolRun/SetupDeckCalibration.tsx index 9b0dd34479f..3ec0a03e72c 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupDeckCalibration.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupDeckCalibration.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { Link } from 'react-router-dom' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupFlexPipetteCalibrationItem.tsx b/app/src/organisms/Devices/ProtocolRun/SetupFlexPipetteCalibrationItem.tsx index fda16d0fa2a..e78873c59fa 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupFlexPipetteCalibrationItem.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupFlexPipetteCalibrationItem.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { Flex, @@ -15,7 +15,7 @@ import { } from '@opentrons/shared-data' import { useInstrumentsQuery } from '@opentrons/react-api-client' import { TertiaryButton } from '/app/atoms/buttons' -import { useMostRecentCompletedAnalysis } from '../../LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' import { useStoredProtocolAnalysis } from '/app/resources/analysis' import { PipetteWizardFlows } from '../../PipetteWizardFlows' import { FLOWS } from '../../PipetteWizardFlows/constants' @@ -36,9 +36,7 @@ export function SetupFlexPipetteCalibrationItem({ instrumentsRefetch, }: SetupInstrumentCalibrationItemProps): JSX.Element | null { const { t } = useTranslation(['protocol_setup', 'devices_landing']) - const [showFlexPipetteFlow, setShowFlexPipetteFlow] = React.useState( - false - ) + const [showFlexPipetteFlow, setShowFlexPipetteFlow] = useState(false) const { data: attachedInstruments } = useInstrumentsQuery() const mostRecentAnalysis = useMostRecentCompletedAnalysis(runId) const storedProtocolAnalysis = useStoredProtocolAnalysis(runId) diff --git a/app/src/organisms/Devices/ProtocolRun/SetupGripperCalibrationItem.tsx b/app/src/organisms/Devices/ProtocolRun/SetupGripperCalibrationItem.tsx index 3f75520fcff..1eb23ba091d 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupGripperCalibrationItem.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupGripperCalibrationItem.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { Flex, @@ -31,7 +31,7 @@ export function SetupGripperCalibrationItem({ const [ openWizardFlowType, setOpenWizardFlowType, - ] = React.useState(null) + ] = useState(null) const gripperCalLastModified = gripperData != null diff --git a/app/src/organisms/Devices/ProtocolRun/SetupInstrumentCalibration.tsx b/app/src/organisms/Devices/ProtocolRun/SetupInstrumentCalibration.tsx index e2ff193bcd9..0ce395e0e46 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupInstrumentCalibration.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupInstrumentCalibration.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { @@ -10,15 +9,17 @@ import { TYPOGRAPHY, } from '@opentrons/components' import * as PipetteConstants from '/app/redux/pipettes/constants' -import { getShowPipetteCalibrationWarning } from '../utils' +import { getShowPipetteCalibrationWarning } from '/app/transformations/instruments' import { PipetteRecalibrationWarning } from '../PipetteCard/PipetteRecalibrationWarning' -import { useRunPipetteInfoByMount } from '../hooks' import { useStoredProtocolAnalysis } from '/app/resources/analysis' import { useIsFlex } from '/app/redux-resources/robots' import { SetupPipetteCalibrationItem } from './SetupPipetteCalibrationItem' import { SetupFlexPipetteCalibrationItem } from './SetupFlexPipetteCalibrationItem' import { SetupGripperCalibrationItem } from './SetupGripperCalibrationItem' -import { useMostRecentCompletedAnalysis } from '../../LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { + useRunPipetteInfoByMount, + useMostRecentCompletedAnalysis, +} from '/app/resources/runs' import { useInstrumentsQuery } from '@opentrons/react-api-client' import { isGripperInCommands } from '/app/resources/protocols/utils' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/CurrentOffsetsModal.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/CurrentOffsetsModal.tsx index 6af52e06a3c..c95923e4cb8 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/CurrentOffsetsModal.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/CurrentOffsetsModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import styled from 'styled-components' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' @@ -24,7 +23,7 @@ import { getIsLabwareOffsetCodeSnippetsOn } from '/app/redux/config' import { LabwareOffsetTabs } from '../../../LabwareOffsetTabs' import { OffsetVector } from '/app/molecules/OffsetVector' import { PythonLabwareOffsetSnippet } from '/app/molecules/PythonLabwareOffsetSnippet' -import { getLatestCurrentOffsets } from '../SetupLabwarePositionCheck/utils' +import { getLatestCurrentOffsets } from '/app/transformations/runs' import type { LabwareOffset } from '@opentrons/api-client' import type { diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareListItem.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareListItem.tsx index 0b950e16720..6269be78e83 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareListItem.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareListItem.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import styled, { css } from 'styled-components' @@ -50,10 +50,12 @@ import type { LoadModuleRunTimeCommand, LoadLabwareRunTimeCommand, } from '@opentrons/shared-data' -import type { ModuleRenderInfoForProtocol } from '../../hooks' -import type { LabwareSetupItem } from '/app/transformations/commands' +import type { ModuleRenderInfoForProtocol } from '/app/resources/runs' +import type { + LabwareSetupItem, + NestedLabwareInfo, +} from '/app/transformations/commands' import type { ModuleTypesThatRequireExtraAttention } from '../utils/getModuleTypesThatRequireExtraAttention' -import type { NestedLabwareInfo } from './getNestedLabwareInfo' const LabwareRow = styled.div` display: ${DISPLAY_GRID}; @@ -95,11 +97,11 @@ export function LabwareListItem( const [ secureLabwareModalType, setSecureLabwareModalType, - ] = React.useState(null) + ] = useState(null) const labwareDisplayName = getLabwareDisplayName(definition) const { createLiveCommand } = useCreateLiveCommandMutation() - const [isLatchLoading, setIsLatchLoading] = React.useState(false) - const [isLatchClosed, setIsLatchClosed] = React.useState(false) + const [isLatchLoading, setIsLatchLoading] = useState(false) + const [isLatchClosed, setIsLatchClosed] = useState(false) let slotInfo: string | null = null diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/OffDeckLabwareList.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/OffDeckLabwareList.tsx index 9216a1274da..66853d79579 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/OffDeckLabwareList.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/OffDeckLabwareList.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { SPACING, TYPOGRAPHY, LegacyStyledText } from '@opentrons/components' import { LabwareListItem } from './LabwareListItem' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/SecureLabwareModal.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/SecureLabwareModal.tsx index d9d4ec14388..36c9101ad87 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/SecureLabwareModal.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/SecureLabwareModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { createPortal } from 'react-dom' import snakeCase from 'lodash/snakeCase' import { Trans, useTranslation } from 'react-i18next' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareList.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareList.tsx index 61f836dffd5..647f1543677 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareList.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareList.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { DIRECTION_COLUMN, @@ -7,12 +6,14 @@ import { StyledText, COLORS, } from '@opentrons/components' -import { getLabwareSetupItemGroups } from '/app/transformations/commands' +import { + getLabwareSetupItemGroups, + getNestedLabwareInfo, +} from '/app/transformations/commands' import { LabwareListItem } from './LabwareListItem' -import { getNestedLabwareInfo } from './getNestedLabwareInfo' import type { RunTimeCommand } from '@opentrons/shared-data' -import type { ModuleRenderInfoForProtocol } from '../../hooks' +import type { ModuleRenderInfoForProtocol } from '/app/resources/runs' import type { ModuleTypesThatRequireExtraAttention } from '../utils/getModuleTypesThatRequireExtraAttention' import type { LabwareSetupItem } from '/app/transformations/commands' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareMap.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareMap.tsx index feb3b1be340..0334496fd6b 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareMap.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareMap.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import map from 'lodash/map' import { @@ -18,16 +18,18 @@ import { import { getLabwareSetupItemGroups } from '/app/transformations/commands' import { LabwareInfoOverlay } from '../LabwareInfoOverlay' -import { getLabwareRenderInfo } from '../utils/getLabwareRenderInfo' -import { getProtocolModulesInfo } from '../utils/getProtocolModulesInfo' -import { getStandardDeckViewLayerBlockList } from '../utils/getStandardDeckViewLayerBlockList' +import { + getProtocolModulesInfo, + getLabwareRenderInfo, +} from '/app/transformations/analysis' +import { getStandardDeckViewLayerBlockList } from '/app/local-resources/deck_configuration' import { OffDeckLabwareList } from './OffDeckLabwareList' import type { CompletedProtocolAnalysis, ProtocolAnalysisOutput, } from '@opentrons/shared-data' -import { LabwareStackModal } from './LabwareStackModal' +import { LabwareStackModal } from '/app/molecules/LabwareStackModal' interface SetupLabwareMapProps { runId: string @@ -42,10 +44,8 @@ export function SetupLabwareMap({ const [ labwareStackDetailsLabwareId, setLabwareStackDetailsLabwareId, - ] = React.useState(null) - const [hoverLabwareId, setHoverLabwareId] = React.useState( - null - ) + ] = useState(null) + const [hoverLabwareId, setHoverLabwareId] = useState(null) if (protocolAnalysis == null) return null diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/LabwareListItem.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/LabwareListItem.test.tsx index 172257ecc8e..822b4a41f5a 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/LabwareListItem.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/LabwareListItem.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach, vi, expect } from 'vitest' import { MemoryRouter } from 'react-router-dom' @@ -25,7 +25,7 @@ import type { LoadModuleRunTimeCommand, } from '@opentrons/shared-data' import type { AttachedModule } from '/app/redux/modules/types' -import type { ModuleRenderInfoForProtocol } from '../../../hooks' +import type { ModuleRenderInfoForProtocol } from '/app/resources/runs' vi.mock('../SecureLabwareModal') vi.mock('@opentrons/react-api-client') diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/OffDeckLabwareList.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/OffDeckLabwareList.test.tsx index 163d7e8871b..29ec88b86b5 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/OffDeckLabwareList.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/OffDeckLabwareList.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { screen } from '@testing-library/react' import { describe, it, beforeEach, vi, expect } from 'vitest' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/SecureLabwareModal.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/SecureLabwareModal.test.tsx index d346c397d87..80147006dc1 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/SecureLabwareModal.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/SecureLabwareModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach, vi, expect } from 'vitest' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/SetupLabware.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/SetupLabware.test.tsx index fee5fe31dae..9aa6b7cee22 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/SetupLabware.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/SetupLabware.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach, vi, afterEach, expect } from 'vitest' @@ -7,27 +6,26 @@ import { when } from 'vitest-when' import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' import { useLPCSuccessToast } from '../../../hooks/useLPCSuccessToast' -import { LabwarePositionCheck } from '../../../../LabwarePositionCheck' +import { LabwarePositionCheck } from '/app/organisms/LabwarePositionCheck' import { getModuleTypesThatRequireExtraAttention } from '../../utils/getModuleTypesThatRequireExtraAttention' import { getIsLabwareOffsetCodeSnippetsOn } from '/app/redux/config' +import { SetupLabwareList } from '../SetupLabwareList' +import { SetupLabwareMap } from '../SetupLabwareMap' +import { SetupLabware } from '..' import { - useLPCDisabledReason, + useNotifyRunQuery, useRunCalibrationStatus, useRunHasStarted, + useLPCDisabledReason, useUnmatchedModulesForProtocol, -} from '../../../hooks' -import { SetupLabwareList } from '../SetupLabwareList' -import { SetupLabwareMap } from '../SetupLabwareMap' -import { SetupLabware } from '..' -import { useNotifyRunQuery } from '/app/resources/runs' +} from '/app/resources/runs' vi.mock('../SetupLabwareList') vi.mock('../SetupLabwareMap') -vi.mock('../../../../LabwarePositionCheck') +vi.mock('/app/organisms/LabwarePositionCheck') vi.mock('../../utils/getModuleTypesThatRequireExtraAttention') -vi.mock('../../../../RunTimeControl/hooks') +vi.mock('/app/organisms/RunTimeControl/hooks') vi.mock('/app/redux/config') -vi.mock('../../../hooks') vi.mock('../../../hooks/useLPCSuccessToast') vi.mock('/app/resources/runs') vi.mock('/app/redux-resources/robots') diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/SetupLabwareList.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/SetupLabwareList.test.tsx index f1993f5b1ed..85068d39a1b 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/SetupLabwareList.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/SetupLabwareList.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { describe, it, beforeEach, vi } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/SetupLabwareMap.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/SetupLabwareMap.test.tsx index 288a73b2a6e..832524b2e3b 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/SetupLabwareMap.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/__tests__/SetupLabwareMap.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { MemoryRouter } from 'react-router-dom' import { describe, it, beforeEach, vi, afterEach, expect } from 'vitest' @@ -15,7 +15,7 @@ import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' import { getAttachedProtocolModuleMatches } from '../../../../ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/utils' import { LabwareInfoOverlay } from '../../LabwareInfoOverlay' -import { getLabwareRenderInfo } from '../../utils/getLabwareRenderInfo' +import { getLabwareRenderInfo } from '/app/transformations/analysis' import { SetupLabwareMap } from '../SetupLabwareMap' import type { @@ -42,7 +42,7 @@ vi.mock('@opentrons/shared-data', async importOriginal => { vi.mock('../../../../ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/utils') vi.mock('../../LabwareInfoOverlay') -vi.mock('../../utils/getLabwareRenderInfo') +vi.mock('/app/transformations/analysis/getLabwareRenderInfo') vi.mock('../../utils/getModuleTypesThatRequireExtraAttention') vi.mock('../../../../RunTimeControl/hooks') vi.mock('../../../hooks') diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/index.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/index.tsx index 502d2dd5c8a..38963d79dda 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/index.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import map from 'lodash/map' import { @@ -10,10 +9,12 @@ import { } from '@opentrons/components' import { useToggleGroup } from '/app/molecules/ToggleGroup/useToggleGroup' import { getModuleTypesThatRequireExtraAttention } from '../utils/getModuleTypesThatRequireExtraAttention' -import { useMostRecentCompletedAnalysis } from '../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { + useMostRecentCompletedAnalysis, + useModuleRenderInfoForProtocolById, +} from '/app/resources/runs' import { useIsFlex } from '/app/redux-resources/robots' import { useStoredProtocolAnalysis } from '/app/resources/analysis' -import { useModuleRenderInfoForProtocolById } from '../../hooks' import { SetupLabwareMap } from './SetupLabwareMap' import { SetupLabwareList } from './SetupLabwareList' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/CurrentOffsetsTable.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/CurrentOffsetsTable.tsx index 6ea34c38386..49ee315b25a 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/CurrentOffsetsTable.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/CurrentOffsetsTable.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import styled from 'styled-components' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/HowLPCWorksModal.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/HowLPCWorksModal.tsx index 55397d219eb..a674b98b7b2 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/HowLPCWorksModal.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/HowLPCWorksModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/__tests__/CurrentOffsetsTable.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/__tests__/CurrentOffsetsTable.test.tsx index 69e0b7ed44f..009b303f70b 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/__tests__/CurrentOffsetsTable.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/__tests__/CurrentOffsetsTable.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, beforeEach, vi, expect, afterEach } from 'vitest' import { screen } from '@testing-library/react' @@ -11,17 +11,17 @@ import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' import { getIsLabwareOffsetCodeSnippetsOn } from '/app/redux/config' import { LabwarePositionCheck } from '../../../../LabwarePositionCheck' -import { useLPCDisabledReason } from '../../../hooks' -import { getLatestCurrentOffsets } from '../utils' +import { useLPCDisabledReason } from '/app/resources/runs' +import { getLatestCurrentOffsets } from '/app/transformations/runs' import { CurrentOffsetsTable } from '../CurrentOffsetsTable' import type { CompletedProtocolAnalysis } from '@opentrons/shared-data' import type { LabwareOffset } from '@opentrons/api-client' -vi.mock('../../../hooks') +vi.mock('/app/resources/runs') vi.mock('../../../../LabwarePositionCheck') vi.mock('/app/redux/config') -vi.mock('../utils') +vi.mock('/app/transformations/runs') vi.mock('@opentrons/shared-data', async importOriginal => { const actual = await importOriginal() diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/__tests__/HowLPCWorksModal.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/__tests__/HowLPCWorksModal.test.tsx index cb9224e132f..5dd3d15db69 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/__tests__/HowLPCWorksModal.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/__tests__/HowLPCWorksModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/__tests__/SetupLabwarePositionCheck.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/__tests__/SetupLabwarePositionCheck.test.tsx index 59b78ba440c..104cdc64035 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/__tests__/SetupLabwarePositionCheck.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/__tests__/SetupLabwarePositionCheck.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { when } from 'vitest-when' import { MemoryRouter } from 'react-router-dom' import { screen, fireEvent } from '@testing-library/react' @@ -16,14 +15,14 @@ import { useLPCSuccessToast } from '../../../hooks/useLPCSuccessToast' import { getModuleTypesThatRequireExtraAttention } from '../../utils/getModuleTypesThatRequireExtraAttention' import { useLaunchLPC } from '../../../../LabwarePositionCheck/useLaunchLPC' import { getIsLabwareOffsetCodeSnippetsOn } from '/app/redux/config' +import { SetupLabwarePositionCheck } from '..' import { - useLPCDisabledReason, + useNotifyRunQuery, useRunCalibrationStatus, useRunHasStarted, + useLPCDisabledReason, useUnmatchedModulesForProtocol, -} from '../../../hooks' -import { SetupLabwarePositionCheck } from '..' -import { useNotifyRunQuery } from '/app/resources/runs' +} from '/app/resources/runs' import { useRobotType } from '/app/redux-resources/robots' import type { Mock } from 'vitest' @@ -32,7 +31,6 @@ vi.mock('../../../../LabwarePositionCheck/useLaunchLPC') vi.mock('../../utils/getModuleTypesThatRequireExtraAttention') vi.mock('/app/redux-resources/robots') vi.mock('/app/redux/config') -vi.mock('../../../hooks') vi.mock('../../../hooks/useLPCSuccessToast') vi.mock('@opentrons/react-api-client') vi.mock('/app/resources/runs') diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/index.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/index.tsx index c0813f5b36d..a1d6bdfc266 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/index.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { @@ -19,14 +18,16 @@ import { } from '@opentrons/components' import { useProtocolQuery } from '@opentrons/react-api-client' -import { useMostRecentCompletedAnalysis } from '../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' import { useLPCSuccessToast } from '../../hooks/useLPCSuccessToast' -import { useLPCDisabledReason } from '../../hooks' import { useStoredProtocolAnalysis } from '/app/resources/analysis' import { CurrentOffsetsTable } from './CurrentOffsetsTable' import { useLaunchLPC } from '../../../LabwarePositionCheck/useLaunchLPC' -import { getLatestCurrentOffsets } from './utils' -import { useNotifyRunQuery } from '/app/resources/runs' +import { getLatestCurrentOffsets } from '/app/transformations/runs' +import { + useNotifyRunQuery, + useMostRecentCompletedAnalysis, + useLPCDisabledReason, +} from '/app/resources/runs' import { useRobotType } from '/app/redux-resources/robots' import type { LabwareOffset } from '@opentrons/api-client' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/SetupLiquidsList.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLiquids/SetupLiquidsList.tsx index 82bddc9685f..c362101df85 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/SetupLiquidsList.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLiquids/SetupLiquidsList.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' @@ -33,10 +33,13 @@ import { ANALYTICS_OPEN_LIQUID_LABWARE_DETAIL_MODAL, } from '/app/redux/analytics' import { useIsFlex } from '/app/redux-resources/robots' -import { useMostRecentCompletedAnalysis } from '../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' -import { getLocationInfoNames } from '../utils/getLocationInfoNames' -import { LiquidsLabwareDetailsModal } from './LiquidsLabwareDetailsModal' -import { getTotalVolumePerLiquidId, getVolumePerWell } from './utils' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' +import { getLocationInfoNames } from '/app/transformations/commands' +import { LiquidsLabwareDetailsModal } from '/app/organisms/LiquidsLabwareDetailsModal' +import { + getTotalVolumePerLiquidId, + getVolumePerWell, +} from '/app/transformations/analysis' import type { LabwareByLiquidId } from '@opentrons/shared-data' @@ -136,8 +139,8 @@ export function LiquidsListItem(props: LiquidsListItemProps): JSX.Element { isFlex, } = props const { t } = useTranslation('protocol_setup') - const [openItem, setOpenItem] = React.useState(false) - const [liquidDetailsLabwareId, setLiquidDetailsLabwareId] = React.useState< + const [openItem, setOpenItem] = useState(false) + const [liquidDetailsLabwareId, setLiquidDetailsLabwareId] = useState< string | null >(null) const commands = useMostRecentCompletedAnalysis(runId)?.commands diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/SetupLiquidsMap.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLiquids/SetupLiquidsMap.tsx index fa964eb8535..5338a9ce055 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/SetupLiquidsMap.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLiquids/SetupLiquidsMap.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, Fragment } from 'react' import map from 'lodash/map' import isEmpty from 'lodash/isEmpty' @@ -21,11 +21,13 @@ import { } from '@opentrons/shared-data' import { LabwareInfoOverlay } from '../LabwareInfoOverlay' -import { LiquidsLabwareDetailsModal } from './LiquidsLabwareDetailsModal' -import { getWellFillFromLabwareId } from './utils' -import { getLabwareRenderInfo } from '../utils/getLabwareRenderInfo' -import { getStandardDeckViewLayerBlockList } from '../utils/getStandardDeckViewLayerBlockList' -import { getProtocolModulesInfo } from '../utils/getProtocolModulesInfo' +import { LiquidsLabwareDetailsModal } from '/app/organisms/LiquidsLabwareDetailsModal' +import { getStandardDeckViewLayerBlockList } from '/app/local-resources/deck_configuration' +import { + getProtocolModulesInfo, + getLabwareRenderInfo, + getWellFillFromLabwareId, +} from '/app/transformations/analysis' import type { CompletedProtocolAnalysis, @@ -41,8 +43,8 @@ export function SetupLiquidsMap( props: SetupLiquidsMapProps ): JSX.Element | null { const { runId, protocolAnalysis } = props - const [hoverLabwareId, setHoverLabwareId] = React.useState('') - const [liquidDetailsLabwareId, setLiquidDetailsLabwareId] = React.useState< + const [hoverLabwareId, setHoverLabwareId] = useState('') + const [liquidDetailsLabwareId, setLiquidDetailsLabwareId] = useState< string | null >(null) @@ -157,9 +159,7 @@ export function SetupLiquidsMap( ) const labwareHasLiquid = !isEmpty(wellFill) return ( - + { @@ -189,7 +189,7 @@ export function SetupLiquidsMap( labwareHasLiquid={labwareHasLiquid} /> - + ) } )} diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/SetupLiquids.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/SetupLiquids.test.tsx index ce2848ab841..097f30447ee 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/SetupLiquids.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/SetupLiquids.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, beforeEach, vi } from 'vitest' import { screen, fireEvent } from '@testing-library/react' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/SetupLiquidsList.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/SetupLiquidsList.test.tsx index 80a3384dd24..b9f2c60be0f 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/SetupLiquidsList.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/SetupLiquidsList.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { when } from 'vitest-when' import { describe, it, beforeEach, vi, expect } from 'vitest' @@ -16,10 +16,13 @@ import { ANALYTICS_OPEN_LIQUID_LABWARE_DETAIL_MODAL, } from '/app/redux/analytics' import { useIsFlex } from '/app/redux-resources/robots' -import { getLocationInfoNames } from '../../utils/getLocationInfoNames' +import { getLocationInfoNames } from '/app/transformations/commands' import { SetupLiquidsList } from '../SetupLiquidsList' -import { getTotalVolumePerLiquidId, getVolumePerWell } from '../utils' -import { LiquidsLabwareDetailsModal } from '../LiquidsLabwareDetailsModal' +import { + getTotalVolumePerLiquidId, + getVolumePerWell, +} from '/app/transformations/analysis' +import { LiquidsLabwareDetailsModal } from '/app/organisms/LiquidsLabwareDetailsModal' import { useNotifyRunQuery } from '/app/resources/runs' import type { Mock } from 'vitest' @@ -52,10 +55,10 @@ const MOCK_LABWARE_INFO_BY_LIQUID_ID = { ], } -vi.mock('../utils') -vi.mock('../../utils/getLocationInfoNames') +vi.mock('/app/transformations/analysis') +vi.mock('/app/transformations/commands') vi.mock('/app/redux-resources/robots') -vi.mock('../LiquidsLabwareDetailsModal') +vi.mock('/app/organisms/LiquidsLabwareDetailsModal') vi.mock('@opentrons/shared-data', async importOriginal => { const actualSharedData = await importOriginal() return { diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/SetupLiquidsMap.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/SetupLiquidsMap.test.tsx index 4276aa5dd6f..8aea91dbb34 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/SetupLiquidsMap.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/SetupLiquidsMap.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { screen } from '@testing-library/react' import { describe, it, beforeEach, vi, afterEach, expect } from 'vitest' @@ -21,13 +21,15 @@ import { import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' -import { useAttachedModules } from '../../../hooks' +import { useAttachedModules } from '/app/resources/modules' import { LabwareInfoOverlay } from '../../LabwareInfoOverlay' -import { getLabwareRenderInfo } from '../../utils/getLabwareRenderInfo' -import { getStandardDeckViewLayerBlockList } from '../../utils/getStandardDeckViewLayerBlockList' -import { getAttachedProtocolModuleMatches } from '../../../../ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/utils' -import { getProtocolModulesInfo } from '../../utils/getProtocolModulesInfo' -import { mockProtocolModuleInfo } from '../../../../ODD/ProtocolSetup/ProtocolSetupLabware/__fixtures__' +import { getStandardDeckViewLayerBlockList } from '/app/local-resources/deck_configuration' +import { getAttachedProtocolModuleMatches } from '/app/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/utils' +import { + getProtocolModulesInfo, + getLabwareRenderInfo, +} from '/app/transformations/analysis' +import { mockProtocolModuleInfo } from '/app/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/__fixtures__' import { mockFetchModulesSuccessActionPayloadModules } from '/app/redux/modules/__fixtures__' import { SetupLiquidsMap } from '../SetupLiquidsMap' @@ -49,11 +51,9 @@ vi.mock('@opentrons/components', async importOriginal => { vi.mock('@opentrons/components/src/hardware-sim/BaseDeck') vi.mock('../../LabwareInfoOverlay') -vi.mock('../../../hooks') -vi.mock('../utils') -vi.mock('../../utils/getLabwareRenderInfo') +vi.mock('/app/resources/modules') vi.mock('../../../../ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/utils') -vi.mock('../../utils/getProtocolModulesInfo') +vi.mock('/app/transformations/analysis') vi.mock('/app/resources/deck_configuration/utils') vi.mock('@opentrons/shared-data', async importOriginal => { const actual = await importOriginal() diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/index.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLiquids/index.tsx index 3dcbbbe7e13..28a6f84e2d4 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/index.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLiquids/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { JUSTIFY_CENTER, diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/NotConfiguredModal.tsx b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/NotConfiguredModal.tsx index ca5151763fc..90a6f8d5a02 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/NotConfiguredModal.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/NotConfiguredModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import { useUpdateDeckConfigurationMutation } from '@opentrons/react-api-client' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/OT2MultipleModulesHelp.tsx b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/OT2MultipleModulesHelp.tsx index 4661052add2..69386cf29d0 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/OT2MultipleModulesHelp.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/OT2MultipleModulesHelp.tsx @@ -1,8 +1,9 @@ -import * as React from 'react' +import { useState } from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import { ALIGN_FLEX_END, + Banner, Box, DIRECTION_COLUMN, DIRECTION_ROW, @@ -16,7 +17,6 @@ import { TYPOGRAPHY, } from '@opentrons/components' import { getTopPortalEl } from '../../../../App/portal' -import { Banner } from '/app/atoms/Banner' import multipleModuleHelp from '/app/assets/images/Moam_modal_image.png' const HOW_TO_MULTIPLE_MODULES_HREF = @@ -27,7 +27,7 @@ export function OT2MultipleModulesHelp(): JSX.Element { const [ showMultipleModulesModal, setShowMultipleModulesModal, - ] = React.useState(false) + ] = useState(false) const onCloseClick = (): void => { setShowMultipleModulesModal(false) diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/SetupFixtureList.tsx b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/SetupFixtureList.tsx index 192340c4342..28043cc3e55 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/SetupFixtureList.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/SetupFixtureList.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { css } from 'styled-components' import { useTranslation } from 'react-i18next' import { @@ -23,10 +23,13 @@ import { getCutoutDisplayName, getDeckDefFromRobotType, getFixtureDisplayName, + TC_MODULE_LOCATION_OT3, + THERMOCYCLER_V2_FRONT_FIXTURE, + THERMOCYCLER_V2_REAR_FIXTURE, } from '@opentrons/shared-data' import { StatusLabel } from '/app/atoms/StatusLabel' import { TertiaryButton } from '/app/atoms/buttons/TertiaryButton' -import { LocationConflictModal } from './LocationConflictModal' +import { LocationConflictModal } from '/app/organisms/LocationConflictModal' import { NotConfiguredModal } from './NotConfiguredModal' import { getFixtureImage } from './utils' import { DeckFixtureSetupInstructionsModal } from '../../../DeviceDetailsDeckConfiguration/DeckFixtureSetupInstructionsModal' @@ -47,9 +50,32 @@ interface SetupFixtureListProps { export const SetupFixtureList = (props: SetupFixtureListProps): JSX.Element => { const { deckConfigCompatibility, robotName } = props const deckDef = getDeckDefFromRobotType(FLEX_ROBOT_TYPE) + + const hasTwoLabwareThermocyclerConflicts = + deckConfigCompatibility.some( + ({ cutoutFixtureId, missingLabwareDisplayName }) => + cutoutFixtureId === THERMOCYCLER_V2_FRONT_FIXTURE && + missingLabwareDisplayName != null + ) && + deckConfigCompatibility.some( + ({ cutoutFixtureId, missingLabwareDisplayName }) => + cutoutFixtureId === THERMOCYCLER_V2_REAR_FIXTURE && + missingLabwareDisplayName != null + ) + + // if there are two labware conflicts with the thermocycler, don't show the conflict with the thermocycler rear fixture + const filteredDeckConfigCompatibility = deckConfigCompatibility.filter( + ({ cutoutFixtureId }) => { + return ( + !hasTwoLabwareThermocyclerConflicts || + !(cutoutFixtureId === THERMOCYCLER_V2_REAR_FIXTURE) + ) + } + ) + return ( <> - {deckConfigCompatibility.map(cutoutConfigAndCompatibility => { + {filteredDeckConfigCompatibility.map(cutoutConfigAndCompatibility => { // filter out all fixtures that only provide usb module addressable areas // (i.e. everything but MagBlockV1 and StagingAreaWithMagBlockV1) // as they're handled in the Modules Table @@ -89,6 +115,11 @@ export function FixtureListItem({ const isRequiredSingleSlotMissing = missingLabwareDisplayName != null const isConflictingFixtureConfigured = cutoutFixtureId != null && !SINGLE_SLOT_FIXTURES.includes(cutoutFixtureId) + + const isThermocyclerCurrentFixture = + cutoutFixtureId === THERMOCYCLER_V2_FRONT_FIXTURE || + cutoutFixtureId === THERMOCYCLER_V2_REAR_FIXTURE + let statusLabel if (!isCurrentFixtureCompatible) { statusLabel = ( @@ -117,16 +148,15 @@ export function FixtureListItem({ const [ showLocationConflictModal, setShowLocationConflictModal, - ] = React.useState(false) - const [ - showNotConfiguredModal, - setShowNotConfiguredModal, - ] = React.useState(false) + ] = useState(false) + const [showNotConfiguredModal, setShowNotConfiguredModal] = useState( + false + ) const [ showSetupInstructionsModal, setShowSetupInstructionsModal, - ] = React.useState(false) + ] = useState(false) return ( <> @@ -215,7 +245,9 @@ export function FixtureListItem({ - {getCutoutDisplayName(cutoutId)} + {isThermocyclerCurrentFixture && isRequiredSingleSlotMissing + ? TC_MODULE_LOCATION_OT3 + : getCutoutDisplayName(cutoutId)} (false) + const [showModuleSetupModal, setShowModuleSetupModal] = useState( + false + ) const [ showLocationConflictModal, setShowLocationConflictModal, - ] = React.useState(false) + ] = useState(false) - const [showModuleWizard, setShowModuleWizard] = React.useState(false) + const [showModuleWizard, setShowModuleWizard] = useState(false) const { chainLiveCommands, isCommandMutationLoading } = useChainLiveCommands() const [ prepCommandErrorMessage, setPrepCommandErrorMessage, - ] = React.useState('') + ] = useState('') const handleCalibrateClick = (): void => { if (attachedModuleMatch != null) { diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/SetupModulesMap.tsx b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/SetupModulesMap.tsx index cbf220b8d8a..991549d9b62 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/SetupModulesMap.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/SetupModulesMap.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - import { BaseDeck, Box, @@ -13,12 +11,12 @@ import { getSimplestDeckConfigForProtocol, } from '@opentrons/shared-data' -import { useMostRecentCompletedAnalysis } from '../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' -import { getAttachedProtocolModuleMatches } from '../../../ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/utils' -import { ModuleInfo } from '../../ModuleInfo' -import { useAttachedModules } from '../../hooks' -import { getProtocolModulesInfo } from '../utils/getProtocolModulesInfo' -import { getStandardDeckViewLayerBlockList } from '../utils/getStandardDeckViewLayerBlockList' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' +import { getAttachedProtocolModuleMatches } from '/app/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/utils' +import { ModuleInfo } from '/app/molecules/ModuleInfo' +import { useAttachedModules } from '/app/resources/modules' +import { getProtocolModulesInfo } from '/app/transformations/analysis' +import { getStandardDeckViewLayerBlockList } from '/app/local-resources/deck_configuration' import { useNotifyDeckConfigurationQuery } from '/app/resources/deck_configuration' import { useStoredProtocolAnalysis } from '/app/resources/analysis' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/UnMatchedModuleWarning.tsx b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/UnMatchedModuleWarning.tsx index f2151f2626c..6ebc96dc3c8 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/UnMatchedModuleWarning.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/UnMatchedModuleWarning.tsx @@ -1,17 +1,17 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { DIRECTION_COLUMN, + Banner, Flex, SPACING, LegacyStyledText, TYPOGRAPHY, } from '@opentrons/components' -import { Banner } from '/app/atoms/Banner' export const UnMatchedModuleWarning = (): JSX.Element | null => { const { t } = useTranslation('protocol_setup') - const [showBanner, setShowBanner] = React.useState(true) + const [showBanner, setShowBanner] = useState(true) if (!showBanner) return null return ( diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/NotConfiguredModal.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/NotConfiguredModal.test.tsx index ff95c11cee6..9f371faaa64 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/NotConfiguredModal.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/NotConfiguredModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach, vi, expect } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/OT2MultipleModulesHelp.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/OT2MultipleModulesHelp.test.tsx index 3689c8ce2b4..27efa17e029 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/OT2MultipleModulesHelp.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/OT2MultipleModulesHelp.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import '@testing-library/jest-dom/vitest' import { describe, it, beforeEach, vi, expect } from 'vitest' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/SetupFixtureList.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/SetupFixtureList.test.tsx index 7e8f3c1fe56..dd0236ac96d 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/SetupFixtureList.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/SetupFixtureList.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach, vi } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' @@ -13,16 +13,16 @@ import { import { i18n } from '/app/i18n' import { SetupFixtureList } from '../SetupFixtureList' import { NotConfiguredModal } from '../NotConfiguredModal' -import { LocationConflictModal } from '../LocationConflictModal' -import { DeckFixtureSetupInstructionsModal } from '../../../../DeviceDetailsDeckConfiguration/DeckFixtureSetupInstructionsModal' +import { LocationConflictModal } from '/app/organisms/LocationConflictModal' +import { DeckFixtureSetupInstructionsModal } from '/app/organisms/DeviceDetailsDeckConfiguration/DeckFixtureSetupInstructionsModal' import type { CutoutConfigAndCompatibility } from '/app/resources/deck_configuration/hooks' vi.mock('/app/resources/deck_configuration/hooks') -vi.mock('../LocationConflictModal') +vi.mock('/app/organisms/LocationConflictModal') vi.mock('../NotConfiguredModal') vi.mock( - '../../../../DeviceDetailsDeckConfiguration/DeckFixtureSetupInstructionsModal' + '/app/organisms/DeviceDetailsDeckConfiguration/DeckFixtureSetupInstructionsModal' ) const mockDeckConfigCompatibility: CutoutConfigAndCompatibility[] = [ diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/SetupModulesAndDeck.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/SetupModulesAndDeck.test.tsx index 4ba84391917..21926d3c823 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/SetupModulesAndDeck.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/SetupModulesAndDeck.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { describe, it, beforeEach, expect, vi } from 'vitest' import { fireEvent, screen } from '@testing-library/react' @@ -11,22 +11,22 @@ import { } from '/app/resources/deck_configuration/utils' import { useRunHasStarted, - useUnmatchedModulesForProtocol, useModuleCalibrationStatus, -} from '../../../hooks' + useUnmatchedModulesForProtocol, +} from '/app/resources/runs' import { useIsFlex } from '/app/redux-resources/robots' import { SetupModuleAndDeck } from '../index' import { SetupModulesList } from '../SetupModulesList' import { SetupModulesMap } from '../SetupModulesMap' import { SetupFixtureList } from '../SetupFixtureList' -vi.mock('../../../hooks') vi.mock('/app/redux-resources/robots') vi.mock('../SetupModulesList') vi.mock('../SetupModulesMap') vi.mock('../SetupFixtureList') vi.mock('/app/redux/config') vi.mock('/app/resources/deck_configuration/utils') +vi.mock('/app/resources/runs') const MOCK_ROBOT_NAME = 'otie' const MOCK_RUN_ID = '1' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/SetupModulesList.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/SetupModulesList.test.tsx index 0be86e2df3b..3beea61588a 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/SetupModulesList.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/SetupModulesList.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { fireEvent, screen, waitFor } from '@testing-library/react' import { describe, it, beforeEach, expect, vi } from 'vitest' @@ -14,29 +14,28 @@ import { mockThermocycler, } from '/app/redux/modules/__fixtures__' import { useRobot, useIsFlex } from '/app/redux-resources/robots' -import { useChainLiveCommands } from '/app/resources/runs' -import { ModuleSetupModal } from '../../../../ModuleCard/ModuleSetupModal' -import { ModuleWizardFlows } from '../../../../ModuleWizardFlows' import { + useChainLiveCommands, + useRunCalibrationStatus, useModuleRenderInfoForProtocolById, useUnmatchedModulesForProtocol, - useRunCalibrationStatus, -} from '../../../hooks' +} from '/app/resources/runs' +import { ModuleSetupModal } from '/app/organisms/ModuleCard/ModuleSetupModal' +import { ModuleWizardFlows } from '/app/organisms/ModuleWizardFlows' import { OT2MultipleModulesHelp } from '../OT2MultipleModulesHelp' import { UnMatchedModuleWarning } from '../UnMatchedModuleWarning' import { SetupModulesList } from '../SetupModulesList' -import { LocationConflictModal } from '../LocationConflictModal' +import { LocationConflictModal } from '/app/organisms/LocationConflictModal' import type { ModuleModel, ModuleType } from '@opentrons/shared-data' import type { DiscoveredRobot } from '/app/redux/discovery/types' vi.mock('@opentrons/react-api-client') -vi.mock('../../../hooks') vi.mock('/app/redux-resources/robots') -vi.mock('../LocationConflictModal') +vi.mock('/app/organisms/LocationConflictModal') vi.mock('../UnMatchedModuleWarning') -vi.mock('../../../../ModuleCard/ModuleSetupModal') -vi.mock('../../../../ModuleWizardFlows') +vi.mock('/app/organisms/ModuleCard/ModuleSetupModal') +vi.mock('/app/organisms/ModuleWizardFlows') vi.mock('../OT2MultipleModulesHelp') vi.mock('/app/resources/runs') vi.mock('/app/redux/config') diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/SetupModulesMap.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/SetupModulesMap.test.tsx index 8fac42d8ed1..2f9cc646f4e 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/SetupModulesMap.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/SetupModulesMap.test.tsx @@ -1,6 +1,6 @@ /* eslint-disable @typescript-eslint/no-unsafe-argument */ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { MemoryRouter } from 'react-router-dom' import { describe, it, beforeEach, vi, afterEach, expect } from 'vitest' @@ -14,9 +14,9 @@ import { mockThermocycler as mockThermocyclerFixture, mockMagneticModule as mockMagneticModuleFixture, } from '/app/redux/modules/__fixtures__/index' -import { useMostRecentCompletedAnalysis } from '../../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' -import { getAttachedProtocolModuleMatches } from '../../../../ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/utils' -import { ModuleInfo } from '../../../ModuleInfo' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' +import { getAttachedProtocolModuleMatches } from '/app/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/utils' +import { ModuleInfo } from '/app/molecules/ModuleInfo' import { SetupModulesMap } from '../SetupModulesMap' import type { @@ -43,10 +43,10 @@ vi.mock('@opentrons/shared-data', async importOriginal => { inferModuleOrientationFromXCoordinate: vi.fn(), } }) -vi.mock('../../../../LabwarePositionCheck/useMostRecentCompletedAnalysis') -vi.mock('../../../../ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/utils') -vi.mock('../../../ModuleInfo') -vi.mock('../../../hooks') +vi.mock('/app/resources/runs/useMostRecentCompletedAnalysis') +vi.mock('/app/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/utils') +vi.mock('/app/molecules/ModuleInfo') +vi.mock('/app/resources/modules') const render = (props: React.ComponentProps) => { return renderWithProviders( diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/UnMatchedModuleWarning.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/UnMatchedModuleWarning.test.tsx index ada660cd2c8..eb73c0955bf 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/UnMatchedModuleWarning.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/UnMatchedModuleWarning.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/utils.test.ts b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/utils.test.ts index 2e4639a3c98..98e3dade58a 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/utils.test.ts +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/utils.test.ts @@ -1,87 +1,5 @@ import { describe, it, expect } from 'vitest' -import { getFixtureImage, getModuleImage } from '../utils' - -describe('getModuleImage', () => { - it('should render the magnetic module image when the model is a magnetic module gen 1', () => { - const result = getModuleImage('magneticModuleV1') - expect(result).toEqual( - '/app/src/assets/images/magnetic_module_gen_2_transparent.png' - ) - }) - - it('should render the high res magnetic module image when the model is a magnetic module gen 1 high res', () => { - const result = getModuleImage('magneticModuleV1', true) - expect(result).toEqual( - '/app/src/assets/images/modules/magneticModuleV2@3x.png' - ) - }) - - it('should render the magnetic module image when the model is a magnetic module gen 2', () => { - const result = getModuleImage('magneticModuleV2') - expect(result).toEqual( - '/app/src/assets/images/magnetic_module_gen_2_transparent.png' - ) - }) - - it('should render the temperature module image when the model is a temperature module gen 1', () => { - const result = getModuleImage('temperatureModuleV1') - expect(result).toEqual( - '/app/src/assets/images/temp_deck_gen_2_transparent.png' - ) - }) - - it('should render the temperature module image when the model is a temperature module gen 2', () => { - const result = getModuleImage('temperatureModuleV2') - expect(result).toEqual( - '/app/src/assets/images/temp_deck_gen_2_transparent.png' - ) - }) - - it('should render the high res temperature module image when the model is a temperature module high res', () => { - const result = getModuleImage('temperatureModuleV2', true) - expect(result).toEqual( - '/app/src/assets/images/modules/temperatureModuleV2@3x.png' - ) - }) - - it('should render the heater-shaker module image when the model is a heater-shaker module gen 1', () => { - const result = getModuleImage('heaterShakerModuleV1') - expect(result).toEqual( - '/app/src/assets/images/heater_shaker_module_transparent.png' - ) - }) - - it('should render the high res heater-shaker module image when the model is a heater-shaker module gen 1 high res', () => { - const result = getModuleImage('heaterShakerModuleV1', true) - expect(result).toEqual( - '/app/src/assets/images/modules/heaterShakerModuleV1@3x.png' - ) - }) - - it('should render the thermocycler module image when the model is a thermocycler module gen 1', () => { - const result = getModuleImage('thermocyclerModuleV1') - expect(result).toEqual('/app/src/assets/images/thermocycler_closed.png') - }) - - it('should render the high res thermocycler module image when the model is a thermocycler module gen 1 high res', () => { - const result = getModuleImage('thermocyclerModuleV1', true) - expect(result).toEqual( - '/app/src/assets/images/modules/thermocyclerModuleV1@3x.png' - ) - }) - - it('should render the thermocycler module image when the model is a thermocycler module gen 2', () => { - const result = getModuleImage('thermocyclerModuleV2') - expect(result).toEqual( - '/app/src/assets/images/thermocycler_gen_2_closed.png' - ) - }) - - it('should render the magnetic block v1 image when the module is magneticBlockV1', () => { - const result = getModuleImage('magneticBlockV1') - expect(result).toEqual('/app/src/assets/images/magnetic_block_gen_1.png') - }) -}) +import { getFixtureImage } from '../utils' describe('getFixtureImage', () => { it('should render the staging area image', () => { diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/index.tsx b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/index.tsx index 9af9469f322..3d42a645853 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/index.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { @@ -26,7 +25,7 @@ import { useRunHasStarted, useUnmatchedModulesForProtocol, useModuleCalibrationStatus, -} from '../../hooks' +} from '/app/resources/runs' import { SetupModulesMap } from './SetupModulesMap' import { SetupModulesList } from './SetupModulesList' import { SetupFixtureList } from './SetupFixtureList' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/utils.ts b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/utils.ts index e3c4329ca44..a6b47b7ff9d 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/utils.ts +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/utils.ts @@ -11,14 +11,8 @@ import { WASTE_CHUTE_STAGING_AREA_FIXTURES, } from '@opentrons/shared-data' -import magneticModule from '/app/assets/images/magnetic_module_gen_2_transparent.png' import temperatureModule from '/app/assets/images/temp_deck_gen_2_transparent.png' -import thermoModuleGen1 from '/app/assets/images/thermocycler_closed.png' import heaterShakerModule from '/app/assets/images/heater_shaker_module_transparent.png' -import magneticModuleHighRes from '/app/assets/images/modules/magneticModuleV2@3x.png' -import temperatureModuleHighRes from '/app/assets/images/modules/temperatureModuleV2@3x.png' -import thermoModuleGen1HighRes from '/app/assets/images/modules/thermocyclerModuleV1@3x.png' -import heaterShakerModuleHighRes from '/app/assets/images/modules/heaterShakerModuleV1@3x.png' import thermoModuleGen2 from '/app/assets/images/thermocycler_gen_2_closed.png' import magneticBlockGen1 from '/app/assets/images/magnetic_block_gen_1.png' import stagingAreaMagneticBlockGen1 from '/app/assets/images/staging_area_magnetic_block_gen_1.png' @@ -27,31 +21,7 @@ import stagingArea from '/app/assets/images/staging_area_slot.png' import wasteChute from '/app/assets/images/waste_chute.png' import wasteChuteStagingArea from '/app/assets/images/waste_chute_with_staging_area.png' -import type { CutoutFixtureId, ModuleModel } from '@opentrons/shared-data' - -export function getModuleImage( - model: ModuleModel, - highRes: boolean = false -): string { - switch (model) { - case 'magneticModuleV1': - case 'magneticModuleV2': - return highRes ? magneticModuleHighRes : magneticModule - case 'temperatureModuleV1': - case 'temperatureModuleV2': - return highRes ? temperatureModuleHighRes : temperatureModule - case 'heaterShakerModuleV1': - return highRes ? heaterShakerModuleHighRes : heaterShakerModule - case 'thermocyclerModuleV1': - return highRes ? thermoModuleGen1HighRes : thermoModuleGen1 - case 'thermocyclerModuleV2': - return thermoModuleGen2 - case 'magneticBlockV1': - return magneticBlockGen1 - default: - return 'Error: unknown module model' - } -} +import type { CutoutFixtureId } from '@opentrons/shared-data' export function getFixtureImage(cutoutFixtureId: CutoutFixtureId): string { if (cutoutFixtureId === STAGING_AREA_RIGHT_SLOT_FIXTURE) { diff --git a/app/src/organisms/Devices/ProtocolRun/SetupPipetteCalibrationItem.tsx b/app/src/organisms/Devices/ProtocolRun/SetupPipetteCalibrationItem.tsx index 1e5f271fe57..f42066c77d0 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupPipetteCalibrationItem.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupPipetteCalibrationItem.tsx @@ -1,7 +1,7 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { Link as RRDLink } from 'react-router-dom' import { + Banner, Box, Flex, LegacyTooltip, @@ -19,13 +19,11 @@ import { WRAP, } from '@opentrons/components' import { TertiaryButton } from '/app/atoms/buttons' -import { Banner } from '/app/atoms/Banner' import * as PipetteConstants from '/app/redux/pipettes/constants' import { useDeckCalibrationData } from '../hooks' import { SetupCalibrationItem } from './SetupCalibrationItem' -import type { Mount } from '/app/redux/pipettes/types' -import type { PipetteInfo } from '../hooks' +import type { Mount, PipetteInfo } from '/app/redux/pipettes' const inexactPipetteSupportArticle = 'https://support.opentrons.com/s/article/GEN2-pipette-compatibility' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupRobotCalibration.tsx b/app/src/organisms/Devices/ProtocolRun/SetupRobotCalibration.tsx index ff0e2bd4f99..90745500149 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupRobotCalibration.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupRobotCalibration.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { @@ -16,11 +15,10 @@ import { ANALYTICS_PROCEED_TO_MODULE_SETUP_STEP, ANALYTICS_PROCEED_TO_LABWARE_SETUP_STEP, } from '/app/redux/analytics' -import { useRunHasStarted } from '../hooks' import { SetupDeckCalibration } from './SetupDeckCalibration' import { SetupInstrumentCalibration } from './SetupInstrumentCalibration' import { SetupTipLengthCalibration } from './SetupTipLengthCalibration' -import { useRunStatus } from '/app/resources/runs' +import { useRunStatus, useRunHasStarted } from '/app/resources/runs' import { RUN_STATUS_STOPPED } from '@opentrons/api-client' import { useIsFlex } from '/app/redux-resources/robots' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupStep.tsx b/app/src/organisms/Devices/ProtocolRun/SetupStep.tsx index 4aefec39518..25f2baf1d64 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupStep.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupStep.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { diff --git a/app/src/organisms/Devices/ProtocolRun/SetupTipLengthCalibration.tsx b/app/src/organisms/Devices/ProtocolRun/SetupTipLengthCalibration.tsx index c8d45d15ad3..c01b79c3a34 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupTipLengthCalibration.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupTipLengthCalibration.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { Fragment } from 'react' import { useTranslation } from 'react-i18next' import { @@ -11,7 +11,7 @@ import { } from '@opentrons/components' import * as PipetteConstants from '/app/redux/pipettes/constants' -import { useRunPipetteInfoByMount } from '../hooks' +import { useRunPipetteInfoByMount } from '/app/resources/runs' import { SetupCalibrationItem } from './SetupCalibrationItem' import { SetupTipLengthCalibrationButton } from './SetupTipLengthCalibrationButton' interface SetupTipLengthCalibrationProps { @@ -41,7 +41,7 @@ export function SetupTipLengthCalibration({ return null } else { return ( - + {pipetteInfo.tipRacksForPipette.map((tipRackInfo, index) => { const pipetteNotAttached = pipetteInfo.requestedPipetteMatch === 'incompatible' @@ -71,7 +71,7 @@ export function SetupTipLengthCalibration({ /> ) })} - + ) } })} diff --git a/app/src/organisms/Devices/ProtocolRun/SetupTipLengthCalibrationButton.tsx b/app/src/organisms/Devices/ProtocolRun/SetupTipLengthCalibrationButton.tsx index 2060c0eca87..11a7d65655e 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupTipLengthCalibrationButton.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupTipLengthCalibrationButton.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { @@ -21,11 +20,9 @@ import { import { getLabwareDefURI } from '@opentrons/shared-data' import { TertiaryButton } from '/app/atoms/buttons' -import { - useAttachedPipettes, - useDeckCalibrationData, - useRunHasStarted, -} from '../hooks' +import { useAttachedPipettes } from '/app/resources/instruments' +import { useRunHasStarted } from '/app/resources/runs' +import { useDeckCalibrationData } from '../hooks' import { useDashboardCalibrateTipLength } from '/app/pages/Desktop/Devices/CalibrationDashboard/hooks/useDashboardCalibrateTipLength' import type { Mount } from '@opentrons/components' diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/BackToTopButton.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/BackToTopButton.test.tsx index 8c2c3f79038..0293c0ae5b2 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/BackToTopButton.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/BackToTopButton.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { when } from 'vitest-when' import { MemoryRouter } from 'react-router-dom' diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/EmptySetupStep.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/EmptySetupStep.test.tsx index eee0f6fb90b..673d8b4806c 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/EmptySetupStep.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/EmptySetupStep.test.tsx @@ -1,4 +1,4 @@ -import React from 'react' +import type * as React from 'react' import { describe, it, beforeEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/LabwareInfoOverlay.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/LabwareInfoOverlay.test.tsx index 629c60e9209..0198aa9e448 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/LabwareInfoOverlay.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/LabwareInfoOverlay.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { screen } from '@testing-library/react' import { describe, it, beforeEach, vi, afterEach, expect } from 'vitest' @@ -9,9 +9,9 @@ import { import { nestedTextMatcher, renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' import { useCurrentRun } from '/app/resources/runs' -import { getLabwareLocation } from '../utils/getLabwareLocation' +import { getLabwareLocation } from '/app/transformations/commands' import { LabwareInfoOverlay } from '../LabwareInfoOverlay' -import { getLabwareDefinitionUri } from '../utils/getLabwareDefinitionUri' +import { getLabwareDefinitionUri } from '/app/transformations/protocols' import { useLabwareOffsetForLabware } from '../useLabwareOffsetForLabware' import type { LabwareDefinition2, @@ -20,9 +20,9 @@ import type { } from '@opentrons/shared-data' vi.mock('/app/resources/runs') -vi.mock('../utils/getLabwareLocation') +vi.mock('/app/transformations/commands') vi.mock('../../hooks') -vi.mock('../utils/getLabwareDefinitionUri') +vi.mock('/app/transformations/protocols') vi.mock('../useLabwareOffsetForLabware') vi.mock('@opentrons/shared-data', async importOriginal => { diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunModuleControls.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunModuleControls.test.tsx index d7de570a2cc..5dd1507f893 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunModuleControls.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunModuleControls.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { describe, it, beforeEach, vi, afterEach } from 'vitest' import { screen } from '@testing-library/react' @@ -7,8 +7,8 @@ import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' import { useInstrumentsQuery } from '@opentrons/react-api-client' import { ProtocolRunModuleControls } from '../ProtocolRunModuleControls' -import { ModuleCard } from '../../../ModuleCard' -import { useModuleRenderInfoForProtocolById } from '../../hooks' +import { ModuleCard } from '/app/organisms/ModuleCard' +import { useModuleRenderInfoForProtocolById } from '/app/resources/runs' import { mockMagneticModuleGen2, mockTemperatureModuleGen2, @@ -18,8 +18,8 @@ import { import type { ModuleModel, ModuleType } from '@opentrons/shared-data' vi.mock('@opentrons/react-api-client') -vi.mock('../../../ModuleCard') -vi.mock('../../hooks') +vi.mock('/app/organisms/ModuleCard') +vi.mock('/app/resources/runs') const RUN_ID = 'test123' const mockTempMod = { diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunRuntimeParameters.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunRuntimeParameters.test.tsx index 5e295c6b7a6..59425f6ff6b 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunRuntimeParameters.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunRuntimeParameters.test.tsx @@ -1,12 +1,15 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, beforeEach, afterEach, expect } from 'vitest' import { screen } from '@testing-library/react' import { when } from 'vitest-when' import { InfoScreen } from '@opentrons/components' import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' -import { useMostRecentCompletedAnalysis } from '../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' -import { useNotifyRunQuery, useRunStatus } from '/app/resources/runs' +import { + useMostRecentCompletedAnalysis, + useNotifyRunQuery, + useRunStatus, +} from '/app/resources/runs' import { mockSucceededRun, mockIdleUnstartedRun, @@ -27,7 +30,6 @@ vi.mock('@opentrons/components', async importOriginal => { InfoScreen: vi.fn(), } }) -vi.mock('../../../LabwarePositionCheck/useMostRecentCompletedAnalysis') vi.mock('/app/resources/runs') vi.mock('/app/redux/config') diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunSetup.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunSetup.test.tsx index 9dfd04b716f..64aa0d094ae 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunSetup.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunSetup.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { when } from 'vitest-when' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach, vi, afterEach, expect } from 'vitest' @@ -19,18 +18,19 @@ import { getIsFixtureMismatch, getRequiredDeckConfig, } from '/app/resources/deck_configuration/utils' -import { useMostRecentCompletedAnalysis } from '../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' -import { useDeckConfigurationCompatibility } from '/app/resources/deck_configuration/hooks' -import { useRobot, useIsFlex } from '/app/redux-resources/robots' -import { useStoredProtocolAnalysis } from '/app/resources/analysis' import { - useModuleCalibrationStatus, - useProtocolAnalysisErrors, + useMostRecentCompletedAnalysis, useRunCalibrationStatus, - useRunHasStarted, useRunPipetteInfoByMount, + useNotifyRunQuery, + useRunHasStarted, useUnmatchedModulesForProtocol, -} from '../../hooks' + useModuleCalibrationStatus, + useProtocolAnalysisErrors, +} from '/app/resources/runs' +import { useDeckConfigurationCompatibility } from '/app/resources/deck_configuration/hooks' +import { useRobot, useIsFlex } from '/app/redux-resources/robots' +import { useStoredProtocolAnalysis } from '/app/resources/analysis' import { SetupLabware } from '../SetupLabware' import { SetupRobotCalibration } from '../SetupRobotCalibration' @@ -39,21 +39,25 @@ import { SetupModuleAndDeck } from '../SetupModuleAndDeck' import { EmptySetupStep } from '../EmptySetupStep' import { ProtocolRunSetup } from '../ProtocolRunSetup' import type { MissingSteps } from '../ProtocolRunSetup' -import { useNotifyRunQuery } from '/app/resources/runs' import type * as SharedData from '@opentrons/shared-data' -vi.mock('../../hooks') vi.mock('../SetupLabware') vi.mock('../SetupRobotCalibration') vi.mock('../SetupModuleAndDeck') vi.mock('../SetupLiquids') vi.mock('../EmptySetupStep') -vi.mock('../../../LabwarePositionCheck/useMostRecentCompletedAnalysis') +vi.mock('/app/resources/runs/useNotifyRunQuery') +vi.mock('/app/resources/runs/useMostRecentCompletedAnalysis') +vi.mock('/app/resources/runs/useRunCalibrationStatus') +vi.mock('/app/resources/runs/useRunPipetteInfoByMount') +vi.mock('/app/resources/runs/useRunHasStarted') +vi.mock('/app/resources/runs/useUnmatchedModulesForProtocol') +vi.mock('/app/resources/runs/useModuleCalibrationStatus') +vi.mock('/app/resources/runs/useProtocolAnalysisErrors') vi.mock('/app/redux/config') vi.mock('/app/resources/deck_configuration/utils') vi.mock('/app/resources/deck_configuration/hooks') -vi.mock('/app/resources/runs/useNotifyRunQuery') vi.mock('/app/redux-resources/robots') vi.mock('/app/resources/analysis') vi.mock('@opentrons/shared-data', async importOriginal => { diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupCalibrationItem.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupCalibrationItem.test.tsx index 3ffb0438173..e39e5d7c83c 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupCalibrationItem.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupCalibrationItem.test.tsx @@ -1,16 +1,16 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { when } from 'vitest-when' import { describe, it, beforeEach, vi, afterEach } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' -import { useRunHasStarted } from '../../hooks' -import { formatTimestamp } from '../../utils' +import { useRunHasStarted } from '/app/resources/runs' +import { formatTimestamp } from '/app/transformations/runs' import { SetupCalibrationItem } from '../SetupCalibrationItem' -vi.mock('../../hooks') -vi.mock('../../utils') +vi.mock('/app/resources/runs') +vi.mock('/app/transformations/runs') const RUN_ID = '1' diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupDeckCalibration.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupDeckCalibration.test.tsx index 7b8f54191a1..05e43c3a41f 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupDeckCalibration.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupDeckCalibration.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { when } from 'vitest-when' import { MemoryRouter } from 'react-router-dom' import { describe, it, beforeEach, afterEach, vi, expect } from 'vitest' diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupFlexPipetteCalibrationItem.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupFlexPipetteCalibrationItem.test.tsx index c957030ad73..d178c6fedc6 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupFlexPipetteCalibrationItem.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupFlexPipetteCalibrationItem.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach, vi, afterEach } from 'vitest' @@ -7,16 +7,15 @@ import { useInstrumentsQuery } from '@opentrons/react-api-client' import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' -import { useMostRecentCompletedAnalysis } from '../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' import { PipetteWizardFlows } from '../../../PipetteWizardFlows' import { SetupFlexPipetteCalibrationItem } from '../SetupFlexPipetteCalibrationItem' -import _uncastedModifiedSimpleV6Protocol from '../../hooks/__fixtures__/modifiedSimpleV6.json' +import { modifiedSimpleV6Protocol as _uncastedModifiedSimpleV6Protocol } from '/app/resources/runs/__fixtures__' import type { CompletedProtocolAnalysis } from '@opentrons/shared-data' vi.mock('@opentrons/react-api-client') vi.mock('../../../PipetteWizardFlows') -vi.mock('../../../LabwarePositionCheck/useMostRecentCompletedAnalysis') -vi.mock('../../hooks') +vi.mock('/app/resources/runs') vi.mock('/app/resources/analysis') const RUN_ID = '1' diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupPipetteCalibration.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupPipetteCalibration.test.tsx index 2848c2b578f..ee59da5370b 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupPipetteCalibration.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupPipetteCalibration.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { when } from 'vitest-when' import { describe, it, beforeEach, vi, afterEach, expect } from 'vitest' import { screen } from '@testing-library/react' @@ -6,14 +5,15 @@ import { screen } from '@testing-library/react' import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' import { mockTipRackDefinition } from '/app/redux/custom-labware/__fixtures__' -import { useRunPipetteInfoByMount } from '../../hooks' import { SetupPipetteCalibrationItem } from '../SetupPipetteCalibrationItem' import { SetupInstrumentCalibration } from '../SetupInstrumentCalibration' -import { useNotifyRunQuery } from '/app/resources/runs' +import { + useNotifyRunQuery, + useRunPipetteInfoByMount, +} from '/app/resources/runs' -import type { PipetteInfo } from '../../hooks' +import type { PipetteInfo } from '/app/redux/pipettes' -vi.mock('../../hooks') vi.mock('../SetupPipetteCalibrationItem') vi.mock('/app/redux-resources/robots') vi.mock('/app/resources/runs') diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupPipetteCalibrationItem.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupPipetteCalibrationItem.test.tsx index f2857d5e6e8..effcb32b60f 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupPipetteCalibrationItem.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupPipetteCalibrationItem.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { describe, it, beforeEach, vi, afterEach, expect } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupRobotCalibration.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupRobotCalibration.test.tsx index 5b364dfc99a..8d737f26452 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupRobotCalibration.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupRobotCalibration.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach, vi, afterEach, expect } from 'vitest' @@ -11,7 +11,8 @@ import { } from '/app/redux/analytics' import { useIsFlex } from '/app/redux-resources/robots' import { mockDeckCalData } from '/app/redux/calibration/__fixtures__' -import { useDeckCalibrationData, useRunHasStarted } from '../../hooks' +import { useDeckCalibrationData } from '../../hooks' +import { useRunHasStarted } from '/app/resources/runs' import { SetupDeckCalibration } from '../SetupDeckCalibration' import { SetupInstrumentCalibration } from '../SetupInstrumentCalibration' diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupStep.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupStep.test.tsx index 51be9501e09..705866c51b2 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupStep.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupStep.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach, vi, afterEach, expect } from 'vitest' diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupTipLengthCalibration.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupTipLengthCalibration.test.tsx index daaf804c60a..2c6783f0d0e 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupTipLengthCalibration.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupTipLengthCalibration.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { when } from 'vitest-when' import { describe, it, beforeEach, vi, afterEach, expect } from 'vitest' import { screen } from '@testing-library/react' @@ -6,14 +5,14 @@ import { screen } from '@testing-library/react' import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' import { mockTipRackDefinition } from '/app/redux/custom-labware/__fixtures__' -import { useRunPipetteInfoByMount } from '../../hooks' +import { useRunPipetteInfoByMount } from '/app/resources/runs' import { SetupTipLengthCalibrationButton } from '../SetupTipLengthCalibrationButton' import { SetupTipLengthCalibration } from '../SetupTipLengthCalibration' -import type { PipetteInfo } from '../../hooks' +import type { PipetteInfo } from '/app/redux/pipettes' vi.mock('/app/redux/config') -vi.mock('../../hooks') +vi.mock('/app/resources/runs') vi.mock('../SetupTipLengthCalibrationButton') const ROBOT_NAME = 'otie' diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupTipLengthCalibrationButton.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupTipLengthCalibrationButton.test.tsx index bafe7c5d662..ed636c628d2 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/SetupTipLengthCalibrationButton.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/SetupTipLengthCalibrationButton.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { screen, fireEvent } from '@testing-library/react' import { describe, it, beforeEach, vi, afterEach, expect } from 'vitest' @@ -9,7 +9,8 @@ import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' import { mockDeckCalData } from '/app/redux/calibration/__fixtures__' import { mockTipLengthCalLauncher } from '../../hooks/__fixtures__/taskListFixtures' -import { useDeckCalibrationData, useRunHasStarted } from '../../hooks' +import { useDeckCalibrationData } from '../../hooks' +import { useRunHasStarted } from '/app/resources/runs' import { useDashboardCalibrateTipLength } from '/app/pages/Desktop/Devices/CalibrationDashboard/hooks/useDashboardCalibrateTipLength' import { SetupTipLengthCalibrationButton } from '../SetupTipLengthCalibrationButton' @@ -23,6 +24,7 @@ vi.mock( vi.mock('/app/redux/config') vi.mock('/app/redux/sessions/selectors') vi.mock('../../hooks') +vi.mock('/app/resources/runs') const ROBOT_NAME = 'otie' const RUN_ID = '1' diff --git a/app/src/organisms/Devices/ProtocolRun/useLabwareOffsetForLabware.ts b/app/src/organisms/Devices/ProtocolRun/useLabwareOffsetForLabware.ts index 826dc739421..5a88d626080 100644 --- a/app/src/organisms/Devices/ProtocolRun/useLabwareOffsetForLabware.ts +++ b/app/src/organisms/Devices/ProtocolRun/useLabwareOffsetForLabware.ts @@ -1,12 +1,14 @@ import { getLoadedLabwareDefinitionsByUri } from '@opentrons/shared-data' import { getCurrentOffsetForLabwareInLocation } from './utils/getCurrentOffsetForLabwareInLocation' -import { getLabwareDefinitionUri } from './utils/getLabwareDefinitionUri' -import { getLabwareOffsetLocation } from './utils/getLabwareOffsetLocation' -import { useNotifyRunQuery } from '/app/resources/runs' +import { getLabwareDefinitionUri } from '/app/transformations/protocols' +import { getLabwareOffsetLocation } from '/app/transformations/analysis' +import { + useNotifyRunQuery, + useMostRecentCompletedAnalysis, +} from '/app/resources/runs' import type { LabwareOffset } from '@opentrons/api-client' -import { useMostRecentCompletedAnalysis } from '../../LabwarePositionCheck/useMostRecentCompletedAnalysis' export function useLabwareOffsetForLabware( runId: string, diff --git a/app/src/organisms/Devices/ReachableBanner.tsx b/app/src/organisms/Devices/ReachableBanner.tsx index d729203c5d9..8e278092393 100644 --- a/app/src/organisms/Devices/ReachableBanner.tsx +++ b/app/src/organisms/Devices/ReachableBanner.tsx @@ -1,7 +1,5 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' -import { SPACING } from '@opentrons/components' -import { Banner } from '/app/atoms/Banner' +import { SPACING, Banner } from '@opentrons/components' import { REACHABLE } from '/app/redux/discovery' import type { DiscoveredRobot } from '/app/redux/discovery/types' diff --git a/app/src/organisms/Devices/RecentProtocolRuns.tsx b/app/src/organisms/Devices/RecentProtocolRuns.tsx index 9afce240cf5..6f31f0f7688 100644 --- a/app/src/organisms/Devices/RecentProtocolRuns.tsx +++ b/app/src/organisms/Devices/RecentProtocolRuns.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useAllProtocolsQuery } from '@opentrons/react-api-client' import { diff --git a/app/src/organisms/Devices/RobotCard.tsx b/app/src/organisms/Devices/RobotCard.tsx index bf70a6f2ead..e4598567bb8 100644 --- a/app/src/organisms/Devices/RobotCard.tsx +++ b/app/src/organisms/Devices/RobotCard.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' import { useNavigate } from 'react-router-dom' diff --git a/app/src/organisms/Devices/RobotOverview.tsx b/app/src/organisms/Devices/RobotOverview.tsx index 6a69961a199..b07e5a2a822 100644 --- a/app/src/organisms/Devices/RobotOverview.tsx +++ b/app/src/organisms/Devices/RobotOverview.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/DeviceResetModal.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/DeviceResetModal.tsx index 5b3497f3d89..49a6531c302 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/DeviceResetModal.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/DeviceResetModal.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useEffect } from 'react' import { useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' import last from 'lodash/last' @@ -80,7 +80,7 @@ export function DeviceResetModal({ } } - React.useEffect(() => { + useEffect(() => { if (resetRequestStatus === SUCCESS) closeModal() }, [resetRequestStatus, closeModal]) diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/RenameRobotSlideout.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/RenameRobotSlideout.tsx index 1ae6e89a5ff..5c02dc8eb95 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/RenameRobotSlideout.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/RenameRobotSlideout.tsx @@ -5,6 +5,7 @@ import { useForm, Controller } from 'react-hook-form' import { useTranslation } from 'react-i18next' import { COLORS, + Banner, DIRECTION_COLUMN, Flex, InputField, @@ -21,7 +22,6 @@ import { } from '/app/redux/discovery' import { useTrackEvent, ANALYTICS_RENAME_ROBOT } from '/app/redux/analytics' import { Slideout } from '/app/atoms/Slideout' -import { Banner } from '/app/atoms/Banner' import { useIsFlex } from '/app/redux-resources/robots' import type { Resolver, FieldError } from 'react-hook-form' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/__tests__/DeviceResetModal.test.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/__tests__/DeviceResetModal.test.tsx index a74236888fa..a5a3a91d219 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/__tests__/DeviceResetModal.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/__tests__/DeviceResetModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, expect, beforeEach } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/__tests__/DeviceResetSlideout.test.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/__tests__/DeviceResetSlideout.test.tsx index c42b289f7b0..7cc668503dc 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/__tests__/DeviceResetSlideout.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/__tests__/DeviceResetSlideout.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, expect, beforeEach } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/__tests__/RenameRobotSlideout.test.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/__tests__/RenameRobotSlideout.test.tsx index 7a121176b11..6f7b1b4ab28 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/__tests__/RenameRobotSlideout.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/AdvancedTabSlideouts/__tests__/RenameRobotSlideout.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen, waitFor } from '@testing-library/react' import { describe, it, vi, expect, beforeEach } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/DeviceReset.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/DeviceReset.tsx index 5a6aed28894..2249e453fe6 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/DeviceReset.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/DeviceReset.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/DisplayRobotName.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/DisplayRobotName.tsx index e71a87124b1..c6a380cd6e5 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/DisplayRobotName.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/DisplayRobotName.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/EnableStatusLight.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/EnableStatusLight.tsx index 0c0475d5d3a..4e3c7fcc0f0 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/EnableStatusLight.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/EnableStatusLight.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { @@ -12,7 +11,7 @@ import { LegacyStyledText, } from '@opentrons/components' import { ToggleButton } from '/app/atoms/buttons' -import { useLEDLights } from '../../hooks' +import { useLEDLights } from '/app/resources/robot-settings' interface EnableStatusLightProps { robotName: string diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/FactoryMode.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/FactoryMode.tsx index 89a7b9b0597..14cb3766040 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/FactoryMode.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/FactoryMode.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/GantryHoming.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/GantryHoming.tsx index a47fcb8d537..96ea82a59cf 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/GantryHoming.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/GantryHoming.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useDispatch } from 'react-redux' import { useTranslation } from 'react-i18next' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/LegacySettings.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/LegacySettings.tsx index 8b2a9fb61b2..5ea0ae6e5ec 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/LegacySettings.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/LegacySettings.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useDispatch } from 'react-redux' import { useTranslation } from 'react-i18next' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/OpenJupyterControl.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/OpenJupyterControl.tsx index 5653e750ab4..1fae3e17676 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/OpenJupyterControl.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/OpenJupyterControl.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { ALIGN_CENTER, diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/RobotInformation.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/RobotInformation.tsx index 9e7528bbcdc..05d492261ba 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/RobotInformation.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/RobotInformation.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { Box, diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/RobotServerVersion.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/RobotServerVersion.tsx index 650b4d0bb3c..1353131e794 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/RobotServerVersion.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/RobotServerVersion.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/ShortTrashBin.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/ShortTrashBin.tsx index da928b01721..5bc00476406 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/ShortTrashBin.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/ShortTrashBin.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useDispatch } from 'react-redux' import { useTranslation } from 'react-i18next' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/UpdateRobotSoftware.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/UpdateRobotSoftware.tsx index bb480e9a140..afe1481e68f 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/UpdateRobotSoftware.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/UpdateRobotSoftware.tsx @@ -5,6 +5,7 @@ import { css } from 'styled-components' import { ALIGN_CENTER, + Banner, Box, Flex, JUSTIFY_SPACE_BETWEEN, @@ -22,7 +23,6 @@ import { ExternalLink } from '/app/atoms/Link/ExternalLink' import { TertiaryButton } from '/app/atoms/buttons' import { getRobotUpdateDisplayInfo } from '/app/redux/robot-update' import { useDispatchStartRobotUpdate } from '/app/redux/robot-update/hooks' -import { Banner } from '/app/atoms/Banner' import type { State } from '/app/redux/types' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/UsageSettings.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/UsageSettings.tsx index 491a5d8becd..e8843af6019 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/UsageSettings.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/UsageSettings.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useDispatch } from 'react-redux' import { useTranslation } from 'react-i18next' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/UseOlderAspirateBehavior.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/UseOlderAspirateBehavior.tsx index 3f8bc946976..c3496621f18 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/UseOlderAspirateBehavior.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/UseOlderAspirateBehavior.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useDispatch } from 'react-redux' import { useTranslation } from 'react-i18next' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/DeviceReset.test.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/DeviceReset.test.tsx index 986e966c13d..fae578451ba 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/DeviceReset.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/DeviceReset.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, expect } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/DisplayRobotName.test.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/DisplayRobotName.test.tsx index 8b817df27ba..8a4cc86e293 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/DisplayRobotName.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/DisplayRobotName.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, expect } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/EnableStatusLight.test.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/EnableStatusLight.test.tsx index 38560808cff..2e2cc956bde 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/EnableStatusLight.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/EnableStatusLight.test.tsx @@ -1,14 +1,14 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, expect, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' -import { useLEDLights } from '../../../hooks' +import { useLEDLights } from '/app/resources/robot-settings' import { EnableStatusLight } from '../EnableStatusLight' -vi.mock('../../../hooks') +vi.mock('/app/resources/robot-settings') const ROBOT_NAME = 'otie' const mockToggleLights = vi.fn() diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/GantryHoming.test.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/GantryHoming.test.tsx index 49402e00c97..87c43f494f9 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/GantryHoming.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/GantryHoming.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, expect, beforeEach } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/LegacySettings.test.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/LegacySettings.test.tsx index 7da47c95e67..14237d9e738 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/LegacySettings.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/LegacySettings.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, expect, beforeEach } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/OpenJupyterControl.test.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/OpenJupyterControl.test.tsx index fe323a5667b..57a01e25680 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/OpenJupyterControl.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/OpenJupyterControl.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/RobotInformation.test.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/RobotInformation.test.tsx index 985e8b970f0..51769c2b7e5 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/RobotInformation.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/RobotInformation.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/RobotServerVersion.test.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/RobotServerVersion.test.tsx index face0f2e2c7..07f2877ec43 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/RobotServerVersion.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/RobotServerVersion.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { MemoryRouter } from 'react-router-dom' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/ShortTrashBin.test.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/ShortTrashBin.test.tsx index 8d1ee357b9f..3465535546b 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/ShortTrashBin.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/ShortTrashBin.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/Troubleshooting.test.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/Troubleshooting.test.tsx index 6c690e4b114..659154d22ff 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/Troubleshooting.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/Troubleshooting.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { act, waitFor, screen } from '@testing-library/react' import { when } from 'vitest-when' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/UpdateRobotSoftware.test.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/UpdateRobotSoftware.test.tsx index ab56af2567e..d8845e81d06 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/UpdateRobotSoftware.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/UpdateRobotSoftware.test.tsx @@ -1,5 +1,4 @@ /* eslint-disable testing-library/no-node-access */ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/UsageSettings.test.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/UsageSettings.test.tsx index 1543209d00d..98790f1b3bc 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/UsageSettings.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/UsageSettings.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { screen, fireEvent } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/UseOlderAspirateBehavior.test.tsx b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/UseOlderAspirateBehavior.test.tsx index aeee92b4cd5..a44111c1c11 100644 --- a/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/UseOlderAspirateBehavior.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/AdvancedTab/__tests__/UseOlderAspirateBehavior.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { screen, fireEvent } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/FormModal.tsx b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/FormModal.tsx index 3e62f53acaf..60ce3d2a88e 100644 --- a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/FormModal.tsx +++ b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/FormModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Controller } from 'react-hook-form' import styled, { css } from 'styled-components' diff --git a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/FormRow.tsx b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/FormRow.tsx index 40a67ff2c68..1481d3f40f9 100644 --- a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/FormRow.tsx +++ b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/FormRow.tsx @@ -1,5 +1,5 @@ // presentational components for the wifi connect form -import * as React from 'react' +import type * as React from 'react' import styled from 'styled-components' import { FONT_WEIGHT_SEMIBOLD, SPACING } from '@opentrons/components' diff --git a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/KeyFileField.tsx b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/KeyFileField.tsx index da4e9d4db4b..376048ba420 100644 --- a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/KeyFileField.tsx +++ b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/KeyFileField.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useRef } from 'react' import { SelectField } from '@opentrons/components' import { FormRow } from './FormRow' @@ -53,7 +53,7 @@ export const KeyFileField = (props: KeyFileFieldProps): JSX.Element => { fieldState ) const options = [makeKeyOptions(wifiKeys), ADD_NEW_KEY_OPTION_GROUP] - const uploadKeyRef = React.useRef(null) + const uploadKeyRef = useRef(null) const handleValueChange = (_: string, value: string): void => { if (value === ADD_NEW_KEY_VALUE) { diff --git a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/SecurityField.tsx b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/SecurityField.tsx index 142a5e6d624..c9fa4e0c069 100644 --- a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/SecurityField.tsx +++ b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/SecurityField.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { SelectField } from '@opentrons/components' import { SECURITY_NONE, SECURITY_WPA_PSK } from '../constants' diff --git a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/TextField.tsx b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/TextField.tsx index 4cfa4c0c48c..2242dabea91 100644 --- a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/TextField.tsx +++ b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/TextField.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useReducer } from 'react' import { LegacyInputField, @@ -32,7 +32,7 @@ export const TextField = (props: TextFieldProps): JSX.Element => { field, fieldState ) - const [showPw, toggleShowPw] = React.useReducer(show => !show, false) + const [showPw, toggleShowPw] = useReducer(show => !show, false) const type = isPassword && !showPw ? INPUT_TYPE_PASSWORD : INPUT_TYPE_TEXT return ( diff --git a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/index.tsx b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/index.tsx index c9f726dcb76..3e1c731d33e 100644 --- a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/index.tsx +++ b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ConnectModal/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useForm } from 'react-hook-form' import { useResetFormOnSecurityChange } from './form-state' diff --git a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/DisconnectModal.tsx b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/DisconnectModal.tsx index 251c40a26e0..d5efae7a976 100644 --- a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/DisconnectModal.tsx +++ b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/DisconnectModal.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useEffect } from 'react' import { useTranslation } from 'react-i18next' import { useDispatch, useSelector } from 'react-redux' import last from 'lodash/last' @@ -113,7 +113,7 @@ export const DisconnectModal = ({ disconnectModalBody = t('disconnect_from_wifi_network_failure', { ssid }) } - React.useEffect(() => { + useEffect(() => { if (isDisconnected) { dispatch(clearWifiStatus(robotName)) } diff --git a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ResultModal.tsx b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ResultModal.tsx index c929e4bf14b..6628c35dfc5 100644 --- a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ResultModal.tsx +++ b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/ResultModal.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - import { AlertModal, SpinnerModal } from '@opentrons/components' import * as Copy from './i18n' diff --git a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/SelectSsid/NetworkOptionLabel.tsx b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/SelectSsid/NetworkOptionLabel.tsx index 61a78da004e..0a982be826f 100644 --- a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/SelectSsid/NetworkOptionLabel.tsx +++ b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/SelectSsid/NetworkOptionLabel.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import styled from 'styled-components' import { Icon, FONT_BODY_1_DARK, SPACING } from '@opentrons/components' import { SECURITY_NONE } from '/app/redux/networking' diff --git a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/SelectSsid/index.tsx b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/SelectSsid/index.tsx index 6e9aecd7ee8..b85cc72d563 100644 --- a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/SelectSsid/index.tsx +++ b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/SelectSsid/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { CONTEXT_MENU } from '@opentrons/components' import { SelectField } from '/app/atoms/SelectField' import * as Copy from '../i18n' diff --git a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/__tests__/DisconnectModal.test.tsx b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/__tests__/DisconnectModal.test.tsx index 5a6b46acdc7..7976784878d 100644 --- a/app/src/organisms/Devices/RobotSettings/ConnectNetwork/__tests__/DisconnectModal.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/ConnectNetwork/__tests__/DisconnectModal.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/Devices/RobotSettings/RobotSettingsNetworking.tsx b/app/src/organisms/Devices/RobotSettings/RobotSettingsNetworking.tsx index 0ad19fff922..57acc939d51 100644 --- a/app/src/organisms/Devices/RobotSettings/RobotSettingsNetworking.tsx +++ b/app/src/organisms/Devices/RobotSettings/RobotSettingsNetworking.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import { useSelector, useDispatch } from 'react-redux' @@ -56,9 +56,7 @@ export function RobotSettingsNetworking({ const isRobotBusy = useIsRobotBusy({ poll: true }) const isFlex = useIsFlex(robotName) - const [showDisconnectModal, setShowDisconnectModal] = React.useState( - false - ) + const [showDisconnectModal, setShowDisconnectModal] = useState(false) const canDisconnect = useCanDisconnect(robotName) @@ -90,7 +88,7 @@ export function RobotSettingsNetworking({ useInterval(() => dispatch(fetchStatus(robotName)), STATUS_REFRESH_MS, true) - React.useEffect(() => { + useEffect(() => { updateRobotStatus(isRobotBusy) }, [isRobotBusy, updateRobotStatus]) diff --git a/app/src/organisms/Devices/RobotSettings/SelectNetwork.tsx b/app/src/organisms/Devices/RobotSettings/SelectNetwork.tsx index 31994df4f74..db5a7bc5aad 100644 --- a/app/src/organisms/Devices/RobotSettings/SelectNetwork.tsx +++ b/app/src/organisms/Devices/RobotSettings/SelectNetwork.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { createPortal } from 'react-dom' import { useDispatch, useSelector } from 'react-redux' import last from 'lodash/last' @@ -35,7 +35,7 @@ export const SelectNetwork = ({ const eapOptions = useSelector((state: State) => Networking.getEapOptions(state, robotName) ) - const [changeState, setChangeState] = React.useState({ + const [changeState, setChangeState] = useState({ type: null, }) const dispatch = useDispatch() @@ -53,7 +53,7 @@ export const SelectNetwork = ({ } } - React.useEffect(() => { + useEffect(() => { // if we're connecting to a network, ensure we get the info needed to // populate the configuration forms if (changeState.type === CONNECT || changeState.type === JOIN_OTHER) { diff --git a/app/src/organisms/Devices/RobotSettings/SettingToggle.tsx b/app/src/organisms/Devices/RobotSettings/SettingToggle.tsx index dc3bd18a48a..4407779d888 100644 --- a/app/src/organisms/Devices/RobotSettings/SettingToggle.tsx +++ b/app/src/organisms/Devices/RobotSettings/SettingToggle.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useDispatch } from 'react-redux' import { diff --git a/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/MigrationWarningModal.tsx b/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/MigrationWarningModal.tsx index 38d684555f3..cf41e3efe8a 100644 --- a/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/MigrationWarningModal.tsx +++ b/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/MigrationWarningModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { useTranslation } from 'react-i18next' diff --git a/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/UpdateRobotModal.tsx b/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/UpdateRobotModal.tsx index 0f01f83e33d..7814a0d7d4e 100644 --- a/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/UpdateRobotModal.tsx +++ b/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/UpdateRobotModal.tsx @@ -1,10 +1,11 @@ -import * as React from 'react' +import { useEffect } from 'react' import { useTranslation } from 'react-i18next' import { useDispatch, useSelector } from 'react-redux' import styled, { css } from 'styled-components' import { ALIGN_CENTER, + Banner, BORDERS, DIRECTION_COLUMN, Flex, @@ -30,7 +31,6 @@ import { import { ExternalLink } from '/app/atoms/Link/ExternalLink' import { ReleaseNotes } from '/app/molecules/ReleaseNotes' import { useIsRobotBusy } from '../../hooks' -import { Banner } from '/app/atoms/Banner' import { useDispatchStartRobotUpdate } from '/app/redux/robot-update/hooks' import type { State, Dispatch } from '/app/redux/types' @@ -91,7 +91,7 @@ export function UpdateRobotModal({ disabledReason = updateFromFileDisabledReason else if (isRobotBusy) disabledReason = t('robot_busy_protocol') - React.useEffect(() => { + useEffect(() => { dispatch(robotUpdateChangelogSeen(robotName)) }, [robotName]) diff --git a/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/ViewUpdateModal.tsx b/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/ViewUpdateModal.tsx index 2a243063d28..783d0da6b89 100644 --- a/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/ViewUpdateModal.tsx +++ b/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/ViewUpdateModal.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { createPortal } from 'react-dom' import { useSelector } from 'react-redux' @@ -28,7 +28,7 @@ export function ViewUpdateModal( props: ViewUpdateModalProps ): JSX.Element | null { const { robotName, robot, closeModal } = props - const [showAppUpdateModal, setShowAppUpdateModal] = React.useState(true) + const [showAppUpdateModal, setShowAppUpdateModal] = useState(true) const updateInfo = useSelector((state: State) => getRobotUpdateInfo(state, robotName) @@ -44,10 +44,9 @@ export function ViewUpdateModal( useSelector(getAvailableShellUpdate) ) - const [ - showMigrationWarning, - setShowMigrationWarning, - ] = React.useState(robotSystemType === OT2_BALENA) + const [showMigrationWarning, setShowMigrationWarning] = useState( + robotSystemType === OT2_BALENA + ) const notNowButton = { onClick: closeModal, diff --git a/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/__tests__/RobotUpdateProgressModal.test.tsx b/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/__tests__/RobotUpdateProgressModal.test.tsx index 6d86f67d831..a2139f636bd 100644 --- a/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/__tests__/RobotUpdateProgressModal.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/__tests__/RobotUpdateProgressModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { i18n } from '/app/i18n' import { act, fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/__tests__/UpdateRobotModal.test.tsx b/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/__tests__/UpdateRobotModal.test.tsx index 48419b629da..9eb328092eb 100644 --- a/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/__tests__/UpdateRobotModal.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/__tests__/UpdateRobotModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { createStore } from 'redux' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/__tests__/useRobotUpdateInfo.test.tsx b/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/__tests__/useRobotUpdateInfo.test.tsx index 418199bdc90..2897110aacc 100644 --- a/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/__tests__/useRobotUpdateInfo.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/__tests__/useRobotUpdateInfo.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { renderHook } from '@testing-library/react' import { createStore } from 'redux' import { I18nextProvider } from 'react-i18next' diff --git a/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/index.tsx b/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/index.tsx index 6df07dd09f0..c5802005a82 100644 --- a/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/index.tsx +++ b/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useRef, useEffect, useCallback } from 'react' import { useSelector, useDispatch } from 'react-redux' import NiceModal, { useModal } from '@ebay/nice-modal-react' @@ -30,21 +30,21 @@ export const handleUpdateBuildroot = ( const UpdateBuildroot = NiceModal.create( (props: UpdateBuildrootProps): JSX.Element | null => { const { robot } = props - const hasSeenSessionOnce = React.useRef(false) + const hasSeenSessionOnce = useRef(false) const modal = useModal() - const robotName = React.useRef(robot?.name ?? '') + const robotName = useRef(robot?.name ?? '') const dispatch = useDispatch() const session = useSelector(getRobotUpdateSession) if (!hasSeenSessionOnce.current && session) hasSeenSessionOnce.current = true - React.useEffect(() => { + useEffect(() => { if (robotName.current) { dispatch(setRobotUpdateSeen(robotName.current)) } }, [robotName]) - const ignoreUpdate = React.useCallback(() => { + const ignoreUpdate = useCallback(() => { if (robotName.current) { dispatch(robotUpdateIgnored(robotName.current)) } diff --git a/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/useRobotUpdateInfo.ts b/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/useRobotUpdateInfo.ts index 1d39a17e157..d7a01f7d105 100644 --- a/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/useRobotUpdateInfo.ts +++ b/app/src/organisms/Devices/RobotSettings/UpdateBuildroot/useRobotUpdateInfo.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useMemo, useState, useRef } from 'react' import { useSelector } from 'react-redux' import { getRobotUpdateDownloadProgress } from '/app/redux/robot-update' @@ -12,9 +12,7 @@ export function useRobotUpdateInfo( ): { updateStep: UpdateStep | null; progressPercent: number } { const progressPercent = useFindProgressPercentFrom(robotName, session) - const updateStep = React.useMemo(() => determineUpdateStepFrom(session), [ - session, - ]) + const updateStep = useMemo(() => determineUpdateStepFrom(session), [session]) return { updateStep, @@ -26,11 +24,11 @@ function useFindProgressPercentFrom( robotName: string, session: RobotUpdateSession | null ): number { - const [progressPercent, setProgressPercent] = React.useState(0) - const hasSeenDownloadFileStep = React.useRef(false) - const prevSeenUpdateStep = React.useRef(null) - const prevSeenStepProgress = React.useRef(0) - const currentStepWithProgress = React.useRef(-1) + const [progressPercent, setProgressPercent] = useState(0) + const hasSeenDownloadFileStep = useRef(false) + const prevSeenUpdateStep = useRef(null) + const prevSeenStepProgress = useRef(0) + const currentStepWithProgress = useRef(-1) const downloadProgress = useSelector((state: State) => getRobotUpdateDownloadProgress(state, robotName) diff --git a/app/src/organisms/Devices/RobotSettings/__tests__/RobotSettingsAdvanced.test.tsx b/app/src/organisms/Devices/RobotSettings/__tests__/RobotSettingsAdvanced.test.tsx index 94769e6d161..b2eb4a4dafa 100644 --- a/app/src/organisms/Devices/RobotSettings/__tests__/RobotSettingsAdvanced.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/__tests__/RobotSettingsAdvanced.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { MemoryRouter } from 'react-router-dom' import { when } from 'vitest-when' diff --git a/app/src/organisms/Devices/RobotSettings/__tests__/RobotSettingsFeatureFlags.test.tsx b/app/src/organisms/Devices/RobotSettings/__tests__/RobotSettingsFeatureFlags.test.tsx index f5c1ddc94ca..186148e6b44 100644 --- a/app/src/organisms/Devices/RobotSettings/__tests__/RobotSettingsFeatureFlags.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/__tests__/RobotSettingsFeatureFlags.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { when } from 'vitest-when' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/Devices/RobotSettings/__tests__/RobotSettingsNetworking.test.tsx b/app/src/organisms/Devices/RobotSettings/__tests__/RobotSettingsNetworking.test.tsx index e39ca5d6281..7c37ebd0b45 100644 --- a/app/src/organisms/Devices/RobotSettings/__tests__/RobotSettingsNetworking.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/__tests__/RobotSettingsNetworking.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { when } from 'vitest-when' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/organisms/Devices/RobotStatusHeader.tsx b/app/src/organisms/Devices/RobotStatusHeader.tsx index 51afab298b4..77cad0933ea 100644 --- a/app/src/organisms/Devices/RobotStatusHeader.tsx +++ b/app/src/organisms/Devices/RobotStatusHeader.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { useSelector, useDispatch } from 'react-redux' import { Link, useNavigate } from 'react-router-dom' diff --git a/app/src/organisms/Devices/__tests__/CalibrationStatusBanner.test.tsx b/app/src/organisms/Devices/__tests__/CalibrationStatusBanner.test.tsx index 8e893bb3851..fd50c5185ae 100644 --- a/app/src/organisms/Devices/__tests__/CalibrationStatusBanner.test.tsx +++ b/app/src/organisms/Devices/__tests__/CalibrationStatusBanner.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/Devices/__tests__/ConnectionTroubleshootingModal.test.tsx b/app/src/organisms/Devices/__tests__/ConnectionTroubleshootingModal.test.tsx index 01e73dc9880..d725929a081 100644 --- a/app/src/organisms/Devices/__tests__/ConnectionTroubleshootingModal.test.tsx +++ b/app/src/organisms/Devices/__tests__/ConnectionTroubleshootingModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/Devices/__tests__/DevicesEmptyState.test.tsx b/app/src/organisms/Devices/__tests__/DevicesEmptyState.test.tsx index 75448e8611e..e05d8499550 100644 --- a/app/src/organisms/Devices/__tests__/DevicesEmptyState.test.tsx +++ b/app/src/organisms/Devices/__tests__/DevicesEmptyState.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/Devices/__tests__/EstopBanner.test.tsx b/app/src/organisms/Devices/__tests__/EstopBanner.test.tsx index 788588ab747..7d052568378 100644 --- a/app/src/organisms/Devices/__tests__/EstopBanner.test.tsx +++ b/app/src/organisms/Devices/__tests__/EstopBanner.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/Devices/__tests__/HistoricalProtocolRun.test.tsx b/app/src/organisms/Devices/__tests__/HistoricalProtocolRun.test.tsx index e91517e45e8..070f1c614de 100644 --- a/app/src/organisms/Devices/__tests__/HistoricalProtocolRun.test.tsx +++ b/app/src/organisms/Devices/__tests__/HistoricalProtocolRun.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/Devices/__tests__/HistoricalProtocolRunOverflowMenu.test.tsx b/app/src/organisms/Devices/__tests__/HistoricalProtocolRunOverflowMenu.test.tsx index bd6aa3e5cba..f9d93904709 100644 --- a/app/src/organisms/Devices/__tests__/HistoricalProtocolRunOverflowMenu.test.tsx +++ b/app/src/organisms/Devices/__tests__/HistoricalProtocolRunOverflowMenu.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' import { when } from 'vitest-when' diff --git a/app/src/organisms/Devices/__tests__/InstrumentsAndModules.test.tsx b/app/src/organisms/Devices/__tests__/InstrumentsAndModules.test.tsx index cb9ab747ed3..e3c927524bc 100644 --- a/app/src/organisms/Devices/__tests__/InstrumentsAndModules.test.tsx +++ b/app/src/organisms/Devices/__tests__/InstrumentsAndModules.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { when } from 'vitest-when' import { screen } from '@testing-library/react' import { describe, it, vi, beforeEach, afterEach, expect } from 'vitest' @@ -11,7 +10,6 @@ import { } from '@opentrons/react-api-client' import { i18n } from '/app/i18n' -import { Banner } from '/app/atoms/Banner' import { mockMagneticModule } from '/app/redux/modules/__fixtures__' import { useIsFlex } from '/app/redux-resources/robots' import { useIsRobotViewable, useRunStatuses } from '../hooks' @@ -21,7 +19,7 @@ import { GripperCard } from '../../GripperCard' import { PipetteCard } from '../PipetteCard' import { FlexPipetteCard } from '../PipetteCard/FlexPipetteCard' import { PipetteRecalibrationWarning } from '../PipetteCard/PipetteRecalibrationWarning' -import { getShowPipetteCalibrationWarning } from '../utils' +import { getShowPipetteCalibrationWarning } from '/app/transformations/instruments' import { useIsEstopNotDisengaged } from '/app/resources/devices/hooks/useIsEstopNotDisengaged' import { useCurrentRunId } from '/app/resources/runs' import type * as Components from '@opentrons/components' @@ -42,8 +40,7 @@ vi.mock('../PipetteCard/FlexPipetteCard') vi.mock('../PipetteCard/PipetteRecalibrationWarning') vi.mock('/app/resources/runs') vi.mock('/app/redux-resources/robots') -vi.mock('/app/atoms/Banner') -vi.mock('../utils') +vi.mock('/app/transformations/instruments') vi.mock('/app/resources/devices/hooks/useIsEstopNotDisengaged') const ROBOT_NAME = 'otie' @@ -122,7 +119,9 @@ describe('InstrumentsAndModules', () => { it('renders the protocol loaded banner when protocol is loaded and not terminal state', () => { vi.mocked(useCurrentRunId).mockReturnValue('RUNID') render() - expect(vi.mocked(Banner)).toHaveBeenCalled() + screen.getByText( + 'Robot must be on the network to see connected instruments and modules' + ) }) it('renders 1 pipette card when a 96 channel is attached', () => { when(useIsFlex).calledWith(ROBOT_NAME).thenReturn(true) diff --git a/app/src/organisms/Devices/__tests__/RecentProtocolRuns.test.tsx b/app/src/organisms/Devices/__tests__/RecentProtocolRuns.test.tsx index bb183476da5..3127fab7c1e 100644 --- a/app/src/organisms/Devices/__tests__/RecentProtocolRuns.test.tsx +++ b/app/src/organisms/Devices/__tests__/RecentProtocolRuns.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/Devices/__tests__/RobotCard.test.tsx b/app/src/organisms/Devices/__tests__/RobotCard.test.tsx index e5c09e3ebb8..c5dba81b373 100644 --- a/app/src/organisms/Devices/__tests__/RobotCard.test.tsx +++ b/app/src/organisms/Devices/__tests__/RobotCard.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { when } from 'vitest-when' import { screen } from '@testing-library/react' @@ -21,12 +21,13 @@ import { import { mockConnectableRobot } from '/app/redux/discovery/__fixtures__' import { getRobotUpdateDisplayInfo } from '/app/redux/robot-update' import { getRobotModelByName } from '/app/redux/discovery' +import { useAttachedModules } from '/app/resources/modules' import { HEALTH_STATUS_OK, ROBOT_MODEL_OT2, ROBOT_MODEL_OT3, } from '/app/redux/discovery/constants' -import { useAttachedModules, useAttachedPipettes } from '../hooks' +import { useAttachedPipettes } from '/app/resources/instruments' import { UpdateRobotBanner } from '../../UpdateRobotBanner' import { RobotOverflowMenu } from '../RobotOverflowMenu' import { RobotStatusHeader } from '../RobotStatusHeader' @@ -40,7 +41,8 @@ import type { State } from '/app/redux/types' vi.mock('/app/redux/robot-update/selectors') vi.mock('/app/redux/discovery/selectors') -vi.mock('../hooks') +vi.mock('/app/resources/instruments') +vi.mock('/app/resources/modules') vi.mock('/app/redux-resources/robots') vi.mock('../../UpdateRobotBanner') vi.mock('/app/redux/config') diff --git a/app/src/organisms/Devices/__tests__/RobotOverflowMenu.test.tsx b/app/src/organisms/Devices/__tests__/RobotOverflowMenu.test.tsx index 0a8c56914b7..e2c9bbe5067 100644 --- a/app/src/organisms/Devices/__tests__/RobotOverflowMenu.test.tsx +++ b/app/src/organisms/Devices/__tests__/RobotOverflowMenu.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/Devices/__tests__/RobotOverview.test.tsx b/app/src/organisms/Devices/__tests__/RobotOverview.test.tsx index 8f9a570dd7b..a02b1424923 100644 --- a/app/src/organisms/Devices/__tests__/RobotOverview.test.tsx +++ b/app/src/organisms/Devices/__tests__/RobotOverview.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { when } from 'vitest-when' import { screen, fireEvent } from '@testing-library/react' diff --git a/app/src/organisms/Devices/__tests__/RobotOverviewOverflowMenu.test.tsx b/app/src/organisms/Devices/__tests__/RobotOverviewOverflowMenu.test.tsx index 30fc3a30efa..8484558df0f 100644 --- a/app/src/organisms/Devices/__tests__/RobotOverviewOverflowMenu.test.tsx +++ b/app/src/organisms/Devices/__tests__/RobotOverviewOverflowMenu.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { when } from 'vitest-when' diff --git a/app/src/organisms/Devices/__tests__/RobotStatusHeader.test.tsx b/app/src/organisms/Devices/__tests__/RobotStatusHeader.test.tsx index 627c08bcdfe..465c46cc566 100644 --- a/app/src/organisms/Devices/__tests__/RobotStatusHeader.test.tsx +++ b/app/src/organisms/Devices/__tests__/RobotStatusHeader.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { RUN_STATUS_RUNNING } from '@opentrons/api-client' import { when } from 'vitest-when' diff --git a/app/src/organisms/Devices/__tests__/utils.test.tsx b/app/src/organisms/Devices/__tests__/utils.test.tsx index c14b38bfbfd..9df883b3c5d 100644 --- a/app/src/organisms/Devices/__tests__/utils.test.tsx +++ b/app/src/organisms/Devices/__tests__/utils.test.tsx @@ -1,7 +1,6 @@ import { describe, it, expect } from 'vitest' import '@testing-library/jest-dom/vitest' import { - formatTimestamp, getIs96ChannelPipetteAttached, getOffsetCalibrationForMount, } from '../utils' @@ -15,28 +14,6 @@ import type { FetchPipettesResponsePipette, } from '/app/redux/pipettes/types' -describe('formatTimestamp', () => { - it('should format an ISO 8601 date string', () => { - const date = '2021-03-07T18:44:49.366581+00:00' - - expect(formatTimestamp(date)).toMatch( - /^(\d{2})\/(\d{2})\/(\d{4}) (\d{2}):(\d{2}):(\d{2})$/ - ) - }) - - it('should pass through a non-ISO 8601 date string', () => { - const date = '2/22/2022 1:00' - - expect(formatTimestamp(date)).toEqual(date) - }) - - it('should pass through a non-date string', () => { - const noDate = 'A Protocol For Otie' - - expect(formatTimestamp(noDate)).toEqual(noDate) - }) -}) - describe('getIs96ChannelPipetteAttached hook', () => { it('returns false when there is no pipette attached on the left mount', () => { const result = getIs96ChannelPipetteAttached(null) diff --git a/app/src/organisms/Devices/hooks/__fixtures__/taskListFixtures.ts b/app/src/organisms/Devices/hooks/__fixtures__/taskListFixtures.ts index f21e8eb7294..15e9e28fb0d 100644 --- a/app/src/organisms/Devices/hooks/__fixtures__/taskListFixtures.ts +++ b/app/src/organisms/Devices/hooks/__fixtures__/taskListFixtures.ts @@ -1,5 +1,5 @@ import { vi } from 'vitest' -import { formatTimestamp } from '../../utils' +import { formatTimestamp } from '/app/transformations/runs' import type { Mock } from 'vitest' import type { diff --git a/app/src/organisms/Devices/hooks/__tests__/useCalibrationTaskList.test.tsx b/app/src/organisms/Devices/hooks/__tests__/useCalibrationTaskList.test.tsx index 127011ec054..9d675c77f7e 100644 --- a/app/src/organisms/Devices/hooks/__tests__/useCalibrationTaskList.test.tsx +++ b/app/src/organisms/Devices/hooks/__tests__/useCalibrationTaskList.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { createStore } from 'redux' import { I18nextProvider } from 'react-i18next' import { Provider } from 'react-redux' @@ -12,7 +12,7 @@ import { useCalibrationStatusQuery, } from '@opentrons/react-api-client' import { useCalibrationTaskList } from '../useCalibrationTaskList' -import { useAttachedPipettes } from '..' +import { useAttachedPipettes } from '/app/resources/instruments' import { TASK_COUNT, mockAttachedPipettesResponse, @@ -33,7 +33,7 @@ import { i18n } from '/app/i18n' import type { Store } from 'redux' import type { State } from '/app/redux/types' -vi.mock('../') +vi.mock('/app/resources/instruments') vi.mock('@opentrons/react-api-client') const mockPipOffsetCalLauncher = vi.fn() diff --git a/app/src/organisms/Devices/hooks/__tests__/useDeckCalibrationData.test.tsx b/app/src/organisms/Devices/hooks/__tests__/useDeckCalibrationData.test.tsx index e91bf298088..c08f0e3e1e5 100644 --- a/app/src/organisms/Devices/hooks/__tests__/useDeckCalibrationData.test.tsx +++ b/app/src/organisms/Devices/hooks/__tests__/useDeckCalibrationData.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { vi, it, expect, describe, beforeEach, afterEach } from 'vitest' import { Provider } from 'react-redux' diff --git a/app/src/organisms/Devices/hooks/__tests__/useIsRobotViewable.test.tsx b/app/src/organisms/Devices/hooks/__tests__/useIsRobotViewable.test.tsx index fb6cf312c6c..ff3a6752800 100644 --- a/app/src/organisms/Devices/hooks/__tests__/useIsRobotViewable.test.tsx +++ b/app/src/organisms/Devices/hooks/__tests__/useIsRobotViewable.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, expect, describe, beforeEach, afterEach } from 'vitest' import { when } from 'vitest-when' import { Provider } from 'react-redux' diff --git a/app/src/organisms/Devices/hooks/__tests__/useLights.test.tsx b/app/src/organisms/Devices/hooks/__tests__/useLights.test.tsx index 5f45fbecb23..35fc2437826 100644 --- a/app/src/organisms/Devices/hooks/__tests__/useLights.test.tsx +++ b/app/src/organisms/Devices/hooks/__tests__/useLights.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, expect, describe, beforeEach, afterEach } from 'vitest' import { Provider } from 'react-redux' import { createStore } from 'redux' diff --git a/app/src/organisms/Devices/hooks/__tests__/usePipetteOffsetCalibration.test.tsx b/app/src/organisms/Devices/hooks/__tests__/usePipetteOffsetCalibration.test.tsx index e3f2cf6618f..2c23dcb0f61 100644 --- a/app/src/organisms/Devices/hooks/__tests__/usePipetteOffsetCalibration.test.tsx +++ b/app/src/organisms/Devices/hooks/__tests__/usePipetteOffsetCalibration.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, expect, describe, beforeEach, afterEach } from 'vitest' import { when } from 'vitest-when' import { Provider } from 'react-redux' diff --git a/app/src/organisms/Devices/hooks/__tests__/usePipetteOffsetCalibrations.test.tsx b/app/src/organisms/Devices/hooks/__tests__/usePipetteOffsetCalibrations.test.tsx index 33c33fe27c2..9305b34af38 100644 --- a/app/src/organisms/Devices/hooks/__tests__/usePipetteOffsetCalibrations.test.tsx +++ b/app/src/organisms/Devices/hooks/__tests__/usePipetteOffsetCalibrations.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, expect, describe, beforeEach, afterEach } from 'vitest' import { when } from 'vitest-when' import { renderHook } from '@testing-library/react' diff --git a/app/src/organisms/Devices/hooks/__tests__/useSyncRobotClock.test.tsx b/app/src/organisms/Devices/hooks/__tests__/useSyncRobotClock.test.tsx index 9a7f6c6ab18..02b593fbaab 100644 --- a/app/src/organisms/Devices/hooks/__tests__/useSyncRobotClock.test.tsx +++ b/app/src/organisms/Devices/hooks/__tests__/useSyncRobotClock.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, expect, describe, beforeEach, afterEach } from 'vitest' import { Provider } from 'react-redux' import { createStore } from 'redux' diff --git a/app/src/organisms/Devices/hooks/__tests__/useTipLengthCalibrations.test.tsx b/app/src/organisms/Devices/hooks/__tests__/useTipLengthCalibrations.test.tsx index 4162c310037..7281b009b5c 100644 --- a/app/src/organisms/Devices/hooks/__tests__/useTipLengthCalibrations.test.tsx +++ b/app/src/organisms/Devices/hooks/__tests__/useTipLengthCalibrations.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, expect, describe, beforeEach, afterEach } from 'vitest' import { when } from 'vitest-when' import { QueryClient, QueryClientProvider } from 'react-query' diff --git a/app/src/organisms/Devices/hooks/__tests__/useTrackCreateProtocolRunEvent.test.tsx b/app/src/organisms/Devices/hooks/__tests__/useTrackCreateProtocolRunEvent.test.tsx index 88ec0f8a9fe..b8cf7fd9896 100644 --- a/app/src/organisms/Devices/hooks/__tests__/useTrackCreateProtocolRunEvent.test.tsx +++ b/app/src/organisms/Devices/hooks/__tests__/useTrackCreateProtocolRunEvent.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { createStore } from 'redux' import { Provider } from 'react-redux' import { QueryClient, QueryClientProvider } from 'react-query' diff --git a/app/src/organisms/Devices/hooks/index.ts b/app/src/organisms/Devices/hooks/index.ts index f60c5a29c27..55199ad90ec 100644 --- a/app/src/organisms/Devices/hooks/index.ts +++ b/app/src/organisms/Devices/hooks/index.ts @@ -1,27 +1,13 @@ -export * from './useAttachedModules' -export * from './useAttachedPipetteCalibrations' -export * from './useAttachedPipettes' export * from './useDeckCalibrationData' -export * from './useDeckCalibrationStatus' export * from './useDownloadRunLog' export * from './useCalibrationTaskList' export * from './useIsRobotBusy' export * from './useIsRobotViewable' export * from './useLights' -export * from './useLEDLights' -export * from './useLPCDisabledReason' export * from './useLPCSuccessToast' -export * from './useModuleRenderInfoForProtocolById' -export * from './useModuleCalibrationStatus' export * from './usePipetteOffsetCalibrations' export * from './usePipetteOffsetCalibration' -export * from './useRunCalibrationStatus' -export * from './useRunCreatedAtTimestamp' -export * from './useRunHasStarted' -export * from './useRunPipetteInfoByMount' export * from './useTipLengthCalibrations' -export * from './useUnmatchedModulesForProtocol' -export * from './useProtocolAnalysisErrors' export * from './useRunStartedOrLegacySessionInProgress' export * from './useTrackCreateProtocolRunEvent' export * from './useRunStatuses' diff --git a/app/src/organisms/Devices/hooks/useAttachedPipettesFromInstrumentsQuery.ts b/app/src/organisms/Devices/hooks/useAttachedPipettesFromInstrumentsQuery.ts index 7b73e060e70..84d318ab110 100644 --- a/app/src/organisms/Devices/hooks/useAttachedPipettesFromInstrumentsQuery.ts +++ b/app/src/organisms/Devices/hooks/useAttachedPipettesFromInstrumentsQuery.ts @@ -1,14 +1,12 @@ import { useInstrumentsQuery } from '@opentrons/react-api-client' import { LEFT, RIGHT } from '@opentrons/shared-data' -import { usePipetteModelSpecs } from '/app/resources/instruments/hooks' +import { usePipetteModelSpecs } from '/app/local-resources/instruments' import type { PipetteData } from '@opentrons/api-client' import type { Mount } from '@opentrons/components' import type { PipetteModel } from '@opentrons/shared-data' +import type { PipetteInformation } from '/app/redux/pipettes' -export interface PipetteInformation extends PipetteData { - displayName: string -} export type AttachedPipettesFromInstrumentsQuery = { [mount in Mount]: null | PipetteInformation } diff --git a/app/src/organisms/Devices/hooks/useCalibrationTaskList.ts b/app/src/organisms/Devices/hooks/useCalibrationTaskList.ts index 9d0c8627eec..222ae1ede68 100644 --- a/app/src/organisms/Devices/hooks/useCalibrationTaskList.ts +++ b/app/src/organisms/Devices/hooks/useCalibrationTaskList.ts @@ -7,10 +7,10 @@ import { } from '@opentrons/react-api-client' import { getLabwareDefURI } from '@opentrons/shared-data' -import { useAttachedPipettes } from '.' +import { useAttachedPipettes } from '/app/resources/instruments' import { getDefaultTiprackDefForPipetteName } from '../constants' import { DECK_CAL_STATUS_OK } from '/app/redux/calibration/constants' -import { formatTimestamp } from '../utils' +import { formatTimestamp } from '/app/transformations/runs' import type { PipetteName } from '@opentrons/shared-data' import type { diff --git a/app/src/organisms/Devices/hooks/useDownloadRunLog.ts b/app/src/organisms/Devices/hooks/useDownloadRunLog.ts index 1dd189e83bc..61acbfb12c4 100644 --- a/app/src/organisms/Devices/hooks/useDownloadRunLog.ts +++ b/app/src/organisms/Devices/hooks/useDownloadRunLog.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { getRun, getCommands, getProtocol } from '@opentrons/api-client' import { useHost } from '@opentrons/react-api-client' @@ -13,7 +13,7 @@ export function useDownloadRunLog( ): { downloadRunLog: () => void; isRunLogLoading: boolean } { const { t } = useTranslation('run_details') const host = useHost() - const [isLoading, setIsLoading] = React.useState(false) + const [isLoading, setIsLoading] = useState(false) const { makeToast } = useToaster() diff --git a/app/src/organisms/Devices/hooks/useLights.ts b/app/src/organisms/Devices/hooks/useLights.ts index dcd9aad698f..02ba9a22dbc 100644 --- a/app/src/organisms/Devices/hooks/useLights.ts +++ b/app/src/organisms/Devices/hooks/useLights.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { useLightsQuery, useSetLightsMutation, @@ -9,13 +9,13 @@ export function useLights(): { lightsOn: boolean | null toggleLights: () => void } { - const [lightsOnCache, setLightsOnCache] = React.useState(false) + const [lightsOnCache, setLightsOnCache] = useState(false) const { setLights, data: setLightsData } = useSetLightsMutation() const { data: lightsData } = useLightsQuery({ refetchInterval: LIGHTS_POLL_MS, }) - React.useEffect(() => { + useEffect(() => { if (setLightsData != null) { setLightsOnCache(setLightsData.on) } else if (lightsData != null) { diff --git a/app/src/organisms/Devices/hooks/usePipetteOffsetCalibration.ts b/app/src/organisms/Devices/hooks/usePipetteOffsetCalibration.ts index ff6522f56ed..c441e618647 100644 --- a/app/src/organisms/Devices/hooks/usePipetteOffsetCalibration.ts +++ b/app/src/organisms/Devices/hooks/usePipetteOffsetCalibration.ts @@ -1,4 +1,4 @@ -import React from 'react' +import { useEffect } from 'react' import { useSelector } from 'react-redux' import { @@ -29,7 +29,7 @@ export function usePipetteOffsetCalibration( ) ) - React.useEffect(() => { + useEffect(() => { if (robotName != null) { dispatchRequest(fetchPipetteOffsetCalibrations(robotName)) } diff --git a/app/src/organisms/Devices/hooks/useSyncRobotClock.ts b/app/src/organisms/Devices/hooks/useSyncRobotClock.ts index ddc74247d6d..9553ca4a3cb 100644 --- a/app/src/organisms/Devices/hooks/useSyncRobotClock.ts +++ b/app/src/organisms/Devices/hooks/useSyncRobotClock.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useEffect } from 'react' import { useDispatch } from 'react-redux' import { syncSystemTime } from '/app/redux/robot-admin' @@ -12,7 +12,7 @@ import type { Dispatch } from '/app/redux/types' export function useSyncRobotClock(robotName: string | null): void { const dispatch = useDispatch() - React.useEffect(() => { + useEffect(() => { if (robotName != null) { dispatch(syncSystemTime(robotName)) } diff --git a/app/src/organisms/Devices/utils.ts b/app/src/organisms/Devices/utils.ts index da0b0666280..c9818264c18 100644 --- a/app/src/organisms/Devices/utils.ts +++ b/app/src/organisms/Devices/utils.ts @@ -1,34 +1,9 @@ -import { format, parseISO } from 'date-fns' -import { INCONSISTENT_PIPETTE_OFFSET } from '@opentrons/api-client' import type { FetchPipettesResponseBody, FetchPipettesResponsePipette, Mount, } from '/app/redux/pipettes/types' -import type { - Instruments, - PipetteData, - PipetteOffsetCalibration, -} from '@opentrons/api-client' - -/** - * formats a string if it is in ISO 8601 date format - * @param {string} timestamp ISO date string - * @returns {string} formatted date string - */ -export function formatTimestamp(timestamp: string): string { - // eslint-disable-next-line eqeqeq - return (parseISO(timestamp) as Date | string) != 'Invalid Date' - ? format(parseISO(timestamp), 'MM/dd/yyyy HH:mm:ss') - : timestamp -} - -export function onDeviceDisplayFormatTimestamp(timestamp: string): string { - // eslint-disable-next-line eqeqeq - return (parseISO(timestamp) as Date | string) != 'Invalid Date' - ? format(parseISO(timestamp), 'HH:mm:ss') - : timestamp -} +import type { PipetteOffsetCalibration } from '@opentrons/api-client' export function downloadFile(data: object | string, fileName: string): void { // Create a blob with the data we want to download as a file @@ -74,19 +49,3 @@ export function getOffsetCalibrationForMount( ) } } - -export function getShowPipetteCalibrationWarning( - attachedInstruments?: Instruments -): boolean { - return ( - attachedInstruments?.data.some((i): i is PipetteData => { - const failuresList = - i.ok && i.data.calibratedOffset?.reasonability_check_failures != null - ? i.data.calibratedOffset?.reasonability_check_failures - : [] - if (failuresList.length > 0) { - return failuresList[0]?.kind === INCONSISTENT_PIPETTE_OFFSET - } else return false - }) ?? false - ) -} diff --git a/app/src/organisms/DropTipWizardFlows/ConfirmPosition.tsx b/app/src/organisms/DropTipWizardFlows/ConfirmPosition.tsx index 6742672beca..d146b9210de 100644 --- a/app/src/organisms/DropTipWizardFlows/ConfirmPosition.tsx +++ b/app/src/organisms/DropTipWizardFlows/ConfirmPosition.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { css } from 'styled-components' import { useTranslation } from 'react-i18next' @@ -34,8 +34,8 @@ export interface UseConfirmPositionResult { export function useConfirmPosition( currentStep: DropTipWizardContainerProps['currentStep'] ): UseConfirmPositionResult { - const [showConfirmPosition, setShowConfirmPosition] = React.useState(false) - const [isRobotPipetteMoving, setIsRobotPipetteMoving] = React.useState(false) + const [showConfirmPosition, setShowConfirmPosition] = useState(false) + const [isRobotPipetteMoving, setIsRobotPipetteMoving] = useState(false) const toggleShowConfirmPosition = (): void => { setShowConfirmPosition(!showConfirmPosition) @@ -46,7 +46,7 @@ export function useConfirmPosition( } // NOTE: The useEffect logic is potentially problematic on views that are not steps, but it is not currently. - React.useEffect(() => { + useEffect(() => { if ( currentStep !== POSITION_AND_BLOWOUT && currentStep !== POSITION_AND_DROP_TIP && diff --git a/app/src/organisms/DropTipWizardFlows/DropTipWizard.tsx b/app/src/organisms/DropTipWizardFlows/DropTipWizard.tsx index caa0c6a733d..cb3f0691113 100644 --- a/app/src/organisms/DropTipWizardFlows/DropTipWizard.tsx +++ b/app/src/organisms/DropTipWizardFlows/DropTipWizard.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' @@ -278,7 +278,7 @@ function useInitiateExit(): { isExitInitiated: boolean toggleExitInitiated: () => void } { - const [isExitInitiated, setIsExitInitiated] = React.useState(false) + const [isExitInitiated, setIsExitInitiated] = useState(false) const toggleExitInitiated = (): void => { setIsExitInitiated(true) diff --git a/app/src/organisms/DropTipWizardFlows/DropTipWizardFlows.tsx b/app/src/organisms/DropTipWizardFlows/DropTipWizardFlows.tsx index 6c7c4530af2..477d84dd4aa 100644 --- a/app/src/organisms/DropTipWizardFlows/DropTipWizardFlows.tsx +++ b/app/src/organisms/DropTipWizardFlows/DropTipWizardFlows.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useDropTipRouting, useDropTipWithType } from './hooks' import { DropTipWizard } from './DropTipWizard' @@ -20,7 +20,7 @@ export function useDropTipWizardFlows(): { showDTWiz: boolean toggleDTWiz: () => void } { - const [showDTWiz, setShowDTWiz] = React.useState(false) + const [showDTWiz, setShowDTWiz] = useState(false) const toggleDTWiz = (): void => { setShowDTWiz(!showDTWiz) diff --git a/app/src/organisms/DropTipWizardFlows/DropTipWizardHeader.tsx b/app/src/organisms/DropTipWizardFlows/DropTipWizardHeader.tsx index 551cb2435c5..e883da67d2b 100644 --- a/app/src/organisms/DropTipWizardFlows/DropTipWizardHeader.tsx +++ b/app/src/organisms/DropTipWizardFlows/DropTipWizardHeader.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { useTranslation } from 'react-i18next' import { BEFORE_BEGINNING, BLOWOUT_SUCCESS, DT_ROUTES } from './constants' @@ -70,11 +70,9 @@ export function useSeenBlowoutSuccess({ currentRoute, currentStepIdx, }: UseSeenBlowoutSuccessProps): UseSeenBlowoutSuccessResult { - const [hasSeenBlowoutSuccess, setHasSeenBlowoutSuccess] = React.useState( - false - ) + const [hasSeenBlowoutSuccess, setHasSeenBlowoutSuccess] = useState(false) - React.useEffect(() => { + useEffect(() => { if (currentStep === BLOWOUT_SUCCESS) { setHasSeenBlowoutSuccess(true) } else if (currentStep === BEFORE_BEGINNING) { diff --git a/app/src/organisms/DropTipWizardFlows/ErrorInfo.tsx b/app/src/organisms/DropTipWizardFlows/ErrorInfo.tsx index 4ae0ef5ad60..400a34fce72 100644 --- a/app/src/organisms/DropTipWizardFlows/ErrorInfo.tsx +++ b/app/src/organisms/DropTipWizardFlows/ErrorInfo.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' diff --git a/app/src/organisms/DropTipWizardFlows/ExitConfirmation.tsx b/app/src/organisms/DropTipWizardFlows/ExitConfirmation.tsx index 69f1a3270a7..9c5486624b7 100644 --- a/app/src/organisms/DropTipWizardFlows/ExitConfirmation.tsx +++ b/app/src/organisms/DropTipWizardFlows/ExitConfirmation.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { css } from 'styled-components' diff --git a/app/src/organisms/DropTipWizardFlows/TipsAttachedModal.tsx b/app/src/organisms/DropTipWizardFlows/TipsAttachedModal.tsx index 401944a29ab..5117c520384 100644 --- a/app/src/organisms/DropTipWizardFlows/TipsAttachedModal.tsx +++ b/app/src/organisms/DropTipWizardFlows/TipsAttachedModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import NiceModal, { useModal } from '@ebay/nice-modal-react' import { Trans, useTranslation } from 'react-i18next' diff --git a/app/src/organisms/DropTipWizardFlows/__tests__/DropTipWizard.test.tsx b/app/src/organisms/DropTipWizardFlows/__tests__/DropTipWizard.test.tsx index 891bac931c2..b5fe7e78b37 100644 --- a/app/src/organisms/DropTipWizardFlows/__tests__/DropTipWizard.test.tsx +++ b/app/src/organisms/DropTipWizardFlows/__tests__/DropTipWizard.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, describe, it, expect, beforeEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/DropTipWizardFlows/__tests__/DropTipWizardHeader.test.tsx b/app/src/organisms/DropTipWizardFlows/__tests__/DropTipWizardHeader.test.tsx index aceecbdd723..69325472a2e 100644 --- a/app/src/organisms/DropTipWizardFlows/__tests__/DropTipWizardHeader.test.tsx +++ b/app/src/organisms/DropTipWizardFlows/__tests__/DropTipWizardHeader.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { beforeEach, describe, expect, it, vi } from 'vitest' import { renderHook, screen } from '@testing-library/react' diff --git a/app/src/organisms/DropTipWizardFlows/__tests__/TipsAttachedModal.test.tsx b/app/src/organisms/DropTipWizardFlows/__tests__/TipsAttachedModal.test.tsx index 07acb2c43e2..2eeaf07f70e 100644 --- a/app/src/organisms/DropTipWizardFlows/__tests__/TipsAttachedModal.test.tsx +++ b/app/src/organisms/DropTipWizardFlows/__tests__/TipsAttachedModal.test.tsx @@ -1,4 +1,3 @@ -import React from 'react' import NiceModal from '@ebay/nice-modal-react' import { describe, it, beforeEach, expect, vi } from 'vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/organisms/DropTipWizardFlows/hooks/__tests__/useTipAttachmentStatus.test.tsx b/app/src/organisms/DropTipWizardFlows/hooks/__tests__/useTipAttachmentStatus.test.tsx index 86d6db11a10..6d9d25719d2 100644 --- a/app/src/organisms/DropTipWizardFlows/hooks/__tests__/useTipAttachmentStatus.test.tsx +++ b/app/src/organisms/DropTipWizardFlows/hooks/__tests__/useTipAttachmentStatus.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' import { act, renderHook } from '@testing-library/react' diff --git a/app/src/organisms/DropTipWizardFlows/hooks/errors.tsx b/app/src/organisms/DropTipWizardFlows/hooks/errors.tsx index cdfa5b32c2b..0a7cdd91d84 100644 --- a/app/src/organisms/DropTipWizardFlows/hooks/errors.tsx +++ b/app/src/organisms/DropTipWizardFlows/hooks/errors.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { AlertPrimaryButton, SPACING } from '@opentrons/components' diff --git a/app/src/organisms/DropTipWizardFlows/hooks/useDropTipCommands.ts b/app/src/organisms/DropTipWizardFlows/hooks/useDropTipCommands.ts index fd8047841d0..8a9579962ed 100644 --- a/app/src/organisms/DropTipWizardFlows/hooks/useDropTipCommands.ts +++ b/app/src/organisms/DropTipWizardFlows/hooks/useDropTipCommands.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { useDeleteMaintenanceRunMutation } from '@opentrons/react-api-client' @@ -59,9 +59,9 @@ export function useDropTipCommands({ fixitCommandTypeUtils, }: UseDropTipSetupCommandsParams): UseDropTipCommandsResult { const isFlex = robotType === FLEX_ROBOT_TYPE - const [hasSeenClose, setHasSeenClose] = React.useState(false) - const [jogQueue, setJogQueue] = React.useState Promise>>([]) - const [isJogging, setIsJogging] = React.useState(false) + const [hasSeenClose, setHasSeenClose] = useState(false) + const [jogQueue, setJogQueue] = useState Promise>>([]) + const [isJogging, setIsJogging] = useState(false) const pipetteId = fixitCommandTypeUtils?.pipetteId ?? null const { deleteMaintenanceRun } = useDeleteMaintenanceRunMutation() @@ -189,7 +189,7 @@ export function useDropTipCommands({ } } - React.useEffect(() => { + useEffect(() => { processJogQueue() }, [jogQueue.length, isJogging]) @@ -353,8 +353,6 @@ const buildBlowoutCommands = ( }, ] : [ - ENGAGE_AXES, - UPDATE_PLUNGER_ESTIMATORS, { commandType: 'blowOutInPlace', params: { diff --git a/app/src/organisms/DropTipWizardFlows/hooks/useDropTipMaintenanceRun.tsx b/app/src/organisms/DropTipWizardFlows/hooks/useDropTipMaintenanceRun.tsx index 9b3e5b67762..fcb69ff467d 100644 --- a/app/src/organisms/DropTipWizardFlows/hooks/useDropTipMaintenanceRun.tsx +++ b/app/src/organisms/DropTipWizardFlows/hooks/useDropTipMaintenanceRun.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { useChainMaintenanceCommands, @@ -32,7 +32,7 @@ export function useDropTipMaintenanceRun({ }: UseDropTipMaintenanceRunParams): string | null { const isMaintenanceRunType = issuedCommandsType === 'setup' - const [createdMaintenanceRunId, setCreatedMaintenanceRunId] = React.useState< + const [createdMaintenanceRunId, setCreatedMaintenanceRunId] = useState< string | null >(null) @@ -101,7 +101,7 @@ function useCreateDropTipMaintenanceRun({ }, }) - React.useEffect(() => { + useEffect(() => { if ( issuedCommandsType === 'setup' && mount != null && @@ -140,10 +140,10 @@ function useMonitorMaintenanceRunForDeletion({ const [ monitorMaintenanceRunForDeletion, setMonitorMaintenanceRunForDeletion, - ] = React.useState(false) - const [closedOnce, setClosedOnce] = React.useState(false) + ] = useState(false) + const [closedOnce, setClosedOnce] = useState(false) - React.useEffect(() => { + useEffect(() => { if (isMaintenanceRunType && !closedOnce) { if ( createdMaintenanceRunId !== null && diff --git a/app/src/organisms/DropTipWizardFlows/hooks/useDropTipRouting.tsx b/app/src/organisms/DropTipWizardFlows/hooks/useDropTipRouting.tsx index 50a72417c8b..17e920644fb 100644 --- a/app/src/organisms/DropTipWizardFlows/hooks/useDropTipRouting.tsx +++ b/app/src/organisms/DropTipWizardFlows/hooks/useDropTipRouting.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useMemo, useState, useEffect } from 'react' import head from 'lodash/head' import last from 'lodash/last' @@ -44,18 +44,16 @@ export interface UseDropTipRoutingResult { export function useDropTipRouting( fixitUtils?: FixitCommandTypeUtils ): UseDropTipRoutingResult { - const [initialRoute, initialStep] = React.useMemo( + const [initialRoute, initialStep] = useMemo( () => getInitialRouteAndStep(fixitUtils), [] ) - const [dropTipFlowsMap, setDropTipFlowsMap] = React.useState( - { - currentRoute: initialRoute as DropTipFlowsRoute, - currentStep: initialStep, - currentStepIdx: 0, - } - ) + const [dropTipFlowsMap, setDropTipFlowsMap] = useState({ + currentRoute: initialRoute as DropTipFlowsRoute, + currentStep: initialStep, + currentStepIdx: 0, + }) useReportMap(dropTipFlowsMap, fixitUtils) @@ -186,7 +184,7 @@ export function useReportMap( ): void { const { currentStep, currentRoute } = map - React.useEffect(() => { + useEffect(() => { if (fixitUtils != null) { fixitUtils.reportMap({ route: currentRoute, step: currentStep }) } diff --git a/app/src/organisms/DropTipWizardFlows/hooks/useDropTipWithType.ts b/app/src/organisms/DropTipWizardFlows/hooks/useDropTipWithType.ts index cb7064ac722..df783ca901f 100644 --- a/app/src/organisms/DropTipWizardFlows/hooks/useDropTipWithType.ts +++ b/app/src/organisms/DropTipWizardFlows/hooks/useDropTipWithType.ts @@ -1,5 +1,5 @@ // This is the main unifying function for maintenanceRun and fixit type flows. -import * as React from 'react' +import { useState, useEffect } from 'react' import { useDropTipCommandErrors } from '.' import { useDropTipMaintenanceRun } from './useDropTipMaintenanceRun' @@ -78,9 +78,7 @@ function useErrorDetails(): { errorDetails: ErrorDetails | null setErrorDetails: (errorDetails: SetRobotErrorDetailsParams) => void } { - const [errorDetails, setErrorDetails] = React.useState( - null - ) + const [errorDetails, setErrorDetails] = useState(null) const setRobustErrorDetails = useDropTipCommandErrors(setErrorDetails) return { errorDetails, setErrorDetails: setRobustErrorDetails } @@ -98,7 +96,7 @@ function useIsExitingDT( isExiting: boolean toggleIsExiting: () => void } { - const [isExiting, setIsExiting] = React.useState(false) + const [isExiting, setIsExiting] = useState(false) const toggleIsExiting = (): void => { setIsExiting(!isExiting) @@ -120,7 +118,7 @@ function useRegisterPipetteFixitType({ chainRunCommands, fixitCommandTypeUtils, }: UseRegisterPipetteFixitType): void { - React.useEffect(() => { + useEffect(() => { if (issuedCommandsType === 'fixit') { const command = buildLoadPipetteCommand( instrumentModelSpecs.name, diff --git a/app/src/organisms/DropTipWizardFlows/hooks/useTipAttachmentStatus/index.ts b/app/src/organisms/DropTipWizardFlows/hooks/useTipAttachmentStatus/index.ts index 80d1bb1913c..99d4ea9abd8 100644 --- a/app/src/organisms/DropTipWizardFlows/hooks/useTipAttachmentStatus/index.ts +++ b/app/src/organisms/DropTipWizardFlows/hooks/useTipAttachmentStatus/index.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useCallback } from 'react' import head from 'lodash/head' import { useInstrumentsQuery } from '@opentrons/react-api-client' @@ -45,10 +45,8 @@ export interface TipAttachmentStatusResult { export function useTipAttachmentStatus( params: Omit ): TipAttachmentStatusResult { - const [pipettesWithTip, setPipettesWithTip] = React.useState< - PipetteWithTip[] - >([]) - const [initialPipettesCount, setInitialPipettesCount] = React.useState< + const [pipettesWithTip, setPipettesWithTip] = useState([]) + const [initialPipettesCount, setInitialPipettesCount] = useState< number | null >(null) const { data: attachedInstruments } = useInstrumentsQuery({ @@ -59,9 +57,7 @@ export function useTipAttachmentStatus( const areTipsAttached = pipettesWithTip.length > 0 && head(pipettesWithTip)?.specs != null - const determineTipStatus = React.useCallback((): Promise< - PipetteWithTip[] - > => { + const determineTipStatus = useCallback((): Promise => { return getPipettesWithTipAttached({ ...params, attachedInstruments: attachedInstruments ?? null, diff --git a/app/src/organisms/DropTipWizardFlows/shared/DropTipFooterButtons.tsx b/app/src/organisms/DropTipWizardFlows/shared/DropTipFooterButtons.tsx index e2f4bcd65ee..6825146c06f 100644 --- a/app/src/organisms/DropTipWizardFlows/shared/DropTipFooterButtons.tsx +++ b/app/src/organisms/DropTipWizardFlows/shared/DropTipFooterButtons.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' diff --git a/app/src/organisms/DropTipWizardFlows/steps/BeforeBeginning.tsx b/app/src/organisms/DropTipWizardFlows/steps/BeforeBeginning.tsx index 36e9b1f4632..e129a177ef2 100644 --- a/app/src/organisms/DropTipWizardFlows/steps/BeforeBeginning.tsx +++ b/app/src/organisms/DropTipWizardFlows/steps/BeforeBeginning.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { css } from 'styled-components' import { useTranslation } from 'react-i18next' @@ -39,7 +39,7 @@ export const BeforeBeginning = ({ modalStyle, }: DropTipWizardContainerProps): JSX.Element | null => { const { t } = useTranslation('drop_tip_wizard') - const [flowType, setFlowType] = React.useState(null) + const [flowType, setFlowType] = useState(null) const handleProceed = (): void => { if (flowType === 'blowout') { diff --git a/app/src/organisms/DropTipWizardFlows/steps/ChooseLocation.tsx b/app/src/organisms/DropTipWizardFlows/steps/ChooseLocation.tsx index a7915c123ec..79db59f9ca6 100644 --- a/app/src/organisms/DropTipWizardFlows/steps/ChooseLocation.tsx +++ b/app/src/organisms/DropTipWizardFlows/steps/ChooseLocation.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { Trans, useTranslation } from 'react-i18next' import { @@ -27,10 +27,7 @@ export const ChooseLocation = ({ }: DropTipWizardContainerProps): JSX.Element | null => { const { moveToAddressableArea } = dropTipCommands const { t } = useTranslation('drop_tip_wizard') - const [ - selectedLocation, - setSelectedLocation, - ] = React.useState() + const [selectedLocation, setSelectedLocation] = useState() const deckDef = getDeckDefFromRobotType(robotType) const handleConfirmPosition = (): void => { diff --git a/app/src/organisms/DropTipWizardFlows/steps/JogToPosition.tsx b/app/src/organisms/DropTipWizardFlows/steps/JogToPosition.tsx index 7277b78aff5..de6b616cf0c 100644 --- a/app/src/organisms/DropTipWizardFlows/steps/JogToPosition.tsx +++ b/app/src/organisms/DropTipWizardFlows/steps/JogToPosition.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' diff --git a/app/src/organisms/DropTipWizardFlows/steps/Success.tsx b/app/src/organisms/DropTipWizardFlows/steps/Success.tsx index 12c56fa48de..5e88a1ce346 100644 --- a/app/src/organisms/DropTipWizardFlows/steps/Success.tsx +++ b/app/src/organisms/DropTipWizardFlows/steps/Success.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { useTranslation } from 'react-i18next' diff --git a/app/src/organisms/EmergencyStop/DesktopEstopMissingModal.stories.tsx b/app/src/organisms/EmergencyStop/DesktopEstopMissingModal.stories.tsx index a1bd81c0654..f8729d48263 100644 --- a/app/src/organisms/EmergencyStop/DesktopEstopMissingModal.stories.tsx +++ b/app/src/organisms/EmergencyStop/DesktopEstopMissingModal.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Provider } from 'react-redux' import { createStore } from 'redux' diff --git a/app/src/organisms/EmergencyStop/DesktopEstopPressedModal.stories.tsx b/app/src/organisms/EmergencyStop/DesktopEstopPressedModal.stories.tsx index 45c7bd121da..5b999ce2bc5 100644 --- a/app/src/organisms/EmergencyStop/DesktopEstopPressedModal.stories.tsx +++ b/app/src/organisms/EmergencyStop/DesktopEstopPressedModal.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Provider } from 'react-redux' import { createStore } from 'redux' import { QueryClient, QueryClientProvider } from 'react-query' diff --git a/app/src/organisms/EmergencyStop/EmergencyStopContext.ts b/app/src/organisms/EmergencyStop/EmergencyStopContext.ts index fd5be4b25a9..d79fae0eec1 100644 --- a/app/src/organisms/EmergencyStop/EmergencyStopContext.ts +++ b/app/src/organisms/EmergencyStop/EmergencyStopContext.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { createContext } from 'react' export interface EmergencyStopContextType { isEmergencyStopModalDismissed: boolean @@ -7,9 +7,7 @@ export interface EmergencyStopContextType { ) => void } -export const EmergencyStopContext = React.createContext( - { - isEmergencyStopModalDismissed: false, - setIsEmergencyStopModalDismissed: () => {}, - } -) +export const EmergencyStopContext = createContext({ + isEmergencyStopModalDismissed: false, + setIsEmergencyStopModalDismissed: () => {}, +}) diff --git a/app/src/organisms/EmergencyStop/EstopMissingModal.tsx b/app/src/organisms/EmergencyStop/EstopMissingModal.tsx index e90ed4bf247..f6f7075b6f2 100644 --- a/app/src/organisms/EmergencyStop/EstopMissingModal.tsx +++ b/app/src/organisms/EmergencyStop/EstopMissingModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { createPortal } from 'react-dom' import { useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' diff --git a/app/src/organisms/EmergencyStop/EstopPressedModal.tsx b/app/src/organisms/EmergencyStop/EstopPressedModal.tsx index 47aee918e90..b36962d4045 100644 --- a/app/src/organisms/EmergencyStop/EstopPressedModal.tsx +++ b/app/src/organisms/EmergencyStop/EstopPressedModal.tsx @@ -4,6 +4,7 @@ import { useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' import { ALIGN_CENTER, + Banner, BORDERS, Chip, COLORS, @@ -24,7 +25,6 @@ import { import { useAcknowledgeEstopDisengageMutation } from '@opentrons/react-api-client' import { getTopPortalEl } from '../../App/portal' -import { Banner } from '/app/atoms/Banner' import { SmallButton } from '/app/atoms/buttons' import { OddModal } from '/app/molecules/OddModal' import { getIsOnDevice } from '/app/redux/config' diff --git a/app/src/organisms/EmergencyStop/EstopTakeover.tsx b/app/src/organisms/EmergencyStop/EstopTakeover.tsx index f7f1977be5d..4792901caa4 100644 --- a/app/src/organisms/EmergencyStop/EstopTakeover.tsx +++ b/app/src/organisms/EmergencyStop/EstopTakeover.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useSelector } from 'react-redux' import { useEstopQuery } from '@opentrons/react-api-client' @@ -22,11 +22,11 @@ interface EstopTakeoverProps { } export function EstopTakeover({ robotName }: EstopTakeoverProps): JSX.Element { - const [estopEngaged, setEstopEngaged] = React.useState(false) + const [estopEngaged, setEstopEngaged] = useState(false) const [ isWaitingForLogicalDisengage, setIsWaitingForLogicalDisengage, - ] = React.useState(false) + ] = useState(false) const { data: estopStatus } = useEstopQuery({ refetchInterval: estopEngaged ? ESTOP_CURRENTLY_ENGAGED_REFETCH_INTERVAL_MS diff --git a/app/src/organisms/EmergencyStop/TouchscreenEstopMissingModal.stories.tsx b/app/src/organisms/EmergencyStop/TouchscreenEstopMissingModal.stories.tsx index 27421ac61c6..7b02f215d42 100644 --- a/app/src/organisms/EmergencyStop/TouchscreenEstopMissingModal.stories.tsx +++ b/app/src/organisms/EmergencyStop/TouchscreenEstopMissingModal.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Provider } from 'react-redux' import { createStore } from 'redux' diff --git a/app/src/organisms/EmergencyStop/TouchscreenEstopPressedModal.stories.tsx b/app/src/organisms/EmergencyStop/TouchscreenEstopPressedModal.stories.tsx index 8e5f30575f9..bfc2c920a34 100644 --- a/app/src/organisms/EmergencyStop/TouchscreenEstopPressedModal.stories.tsx +++ b/app/src/organisms/EmergencyStop/TouchscreenEstopPressedModal.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Provider } from 'react-redux' import { createStore } from 'redux' import { QueryClient, QueryClientProvider } from 'react-query' diff --git a/app/src/organisms/EmergencyStop/__tests__/EstopMissingModal.test.tsx b/app/src/organisms/EmergencyStop/__tests__/EstopMissingModal.test.tsx index bd6fd7bb7fb..c2ce7cea0e1 100644 --- a/app/src/organisms/EmergencyStop/__tests__/EstopMissingModal.test.tsx +++ b/app/src/organisms/EmergencyStop/__tests__/EstopMissingModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import '@testing-library/jest-dom/vitest' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/EmergencyStop/__tests__/EstopPressedModal.test.tsx b/app/src/organisms/EmergencyStop/__tests__/EstopPressedModal.test.tsx index 3393c9072a1..124ea72b3ed 100644 --- a/app/src/organisms/EmergencyStop/__tests__/EstopPressedModal.test.tsx +++ b/app/src/organisms/EmergencyStop/__tests__/EstopPressedModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import '@testing-library/jest-dom/vitest' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/EmergencyStop/__tests__/EstopTakeover.test.tsx b/app/src/organisms/EmergencyStop/__tests__/EstopTakeover.test.tsx index 4c3deece780..f50048f2cf4 100644 --- a/app/src/organisms/EmergencyStop/__tests__/EstopTakeover.test.tsx +++ b/app/src/organisms/EmergencyStop/__tests__/EstopTakeover.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, beforeEach, expect, vi } from 'vitest' import { screen } from '@testing-library/react' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/EmergencyStop/hooks.ts b/app/src/organisms/EmergencyStop/hooks.ts index 6ded6e100ad..775903e0dc9 100644 --- a/app/src/organisms/EmergencyStop/hooks.ts +++ b/app/src/organisms/EmergencyStop/hooks.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useContext } from 'react' import { EmergencyStopContext } from './EmergencyStopContext' import type { EmergencyStopContextType } from './EmergencyStopContext' @@ -7,7 +7,7 @@ export function useEstopContext(): EmergencyStopContextType { const { isEmergencyStopModalDismissed, setIsEmergencyStopModalDismissed, - } = React.useContext(EmergencyStopContext) + } = useContext(EmergencyStopContext) return { isEmergencyStopModalDismissed, diff --git a/app/src/organisms/ErrorRecoveryBanner/__tests__/ErrorRecoveryBanner.test.tsx b/app/src/organisms/ErrorRecoveryBanner/__tests__/ErrorRecoveryBanner.test.tsx index 17e0e5665db..fc234a52629 100644 --- a/app/src/organisms/ErrorRecoveryBanner/__tests__/ErrorRecoveryBanner.test.tsx +++ b/app/src/organisms/ErrorRecoveryBanner/__tests__/ErrorRecoveryBanner.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, beforeEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/ErrorRecoveryBanner/index.tsx b/app/src/organisms/ErrorRecoveryBanner/index.tsx index 6a2d12607cd..f8630d67824 100644 --- a/app/src/organisms/ErrorRecoveryBanner/index.tsx +++ b/app/src/organisms/ErrorRecoveryBanner/index.tsx @@ -1,9 +1,9 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' import { Flex, + Banner, DIRECTION_COLUMN, SPACING, StyledText, @@ -11,7 +11,6 @@ import { import { getUserId } from '/app/redux/config' import { useClientDataRecovery } from '/app/resources/client_data' -import { Banner } from '/app/atoms/Banner' import type { RecoveryIntent } from '/app/resources/client_data' import type { StyleProps } from '@opentrons/components' diff --git a/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryWizard.tsx b/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryWizard.tsx index 1f01ca85028..7fc948f42d9 100644 --- a/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryWizard.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryWizard.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect, useLayoutEffect } from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' @@ -40,11 +40,11 @@ export interface UseERWizardResult { } export function useERWizard(): UseERWizardResult { - const [showERWizard, setShowERWizard] = React.useState(false) + const [showERWizard, setShowERWizard] = useState(false) // Because RunPausedSplash has access to some ER Wiz routes but is not a part of the ER wizard, the splash screen // is the "home" route as opposed to SelectRecoveryOption (accessed by pressing "go back" or "continue" enough times) // when recovery mode has not been launched. - const [hasLaunchedRecovery, setHasLaunchedRecovery] = React.useState(false) + const [hasLaunchedRecovery, setHasLaunchedRecovery] = useState(false) const toggleERWizard = ( isActive: boolean, @@ -64,7 +64,6 @@ export type ErrorRecoveryWizardProps = ErrorRecoveryFlowsProps & ERUtilsResults & { robotType: RobotType isOnDevice: boolean - isDoorOpen: boolean analytics: UseRecoveryAnalyticsResult failedCommand: ReturnType } @@ -95,15 +94,16 @@ export function ErrorRecoveryComponent( const { recoveryMap, hasLaunchedRecovery, - isDoorOpen, + doorStatusUtils, isOnDevice, analytics, } = props + const { isProhibitedDoorOpen } = doorStatusUtils const { route, step } = recoveryMap const { t } = useTranslation('error_recovery') const { showModal, toggleModal } = useErrorDetailsModal() - React.useEffect(() => { + useEffect(() => { if (showModal) { analytics.reportViewErrorDetailsEvent(route, step) } @@ -135,7 +135,7 @@ export function ErrorRecoveryComponent( // TODO(jh, 07-29-24): Make RecoveryDoorOpen render logic equivalent to RecoveryTakeover. Do not nest it in RecoveryWizard. const buildInterventionContent = (): JSX.Element => { - if (isDoorOpen) { + if (isProhibitedDoorOpen) { return } else { return @@ -143,7 +143,7 @@ export function ErrorRecoveryComponent( } const isLargeDesktopStyle = - !isDoorOpen && + !isProhibitedDoorOpen && route === RECOVERY_MAP.DROP_TIP_FLOWS.ROUTE && step !== RECOVERY_MAP.DROP_TIP_FLOWS.STEPS.BEGIN_REMOVAL const desktopType = isLargeDesktopStyle ? 'desktop-large' : 'desktop-small' @@ -218,12 +218,16 @@ export function ErrorRecoveryContent(props: RecoveryContentProps): JSX.Element { return } + const buildManuallyRouteToDoorOpen = (): JSX.Element => { + return + } + switch (props.recoveryMap.route) { case RECOVERY_MAP.OPTION_SELECTION.ROUTE: return buildSelectRecoveryOption() case RECOVERY_MAP.ERROR_WHILE_RECOVERING.ROUTE: return buildRecoveryError() - case RECOVERY_MAP.RETRY_FAILED_COMMAND.ROUTE: + case RECOVERY_MAP.RETRY_STEP.ROUTE: return buildResumeRun() case RECOVERY_MAP.CANCEL_RUN.ROUTE: return buildCancelRun() @@ -248,6 +252,8 @@ export function ErrorRecoveryContent(props: RecoveryContentProps): JSX.Element { case RECOVERY_MAP.ROBOT_PICKING_UP_TIPS.ROUTE: case RECOVERY_MAP.ROBOT_SKIPPING_STEP.ROUTE: return buildRecoveryInProgress() + case RECOVERY_MAP.ROBOT_DOOR_OPEN.ROUTE: + return buildManuallyRouteToDoorOpen() default: return buildSelectRecoveryOption() } @@ -264,14 +270,14 @@ export function useInitialPipetteHome({ routeUpdateActions, }: UseInitialPipetteHomeParams): void { const { homePipetteZAxes } = recoveryCommands - const { setRobotInMotion } = routeUpdateActions + const { handleMotionRouting } = routeUpdateActions // Synchronously set the recovery route to "robot in motion" before initial render to prevent screen flicker on ER launch. - React.useLayoutEffect(() => { + useLayoutEffect(() => { if (hasLaunchedRecovery) { - void setRobotInMotion(true) + void handleMotionRouting(true) .then(() => homePipetteZAxes()) - .finally(() => setRobotInMotion(false)) + .finally(() => handleMotionRouting(false)) } }, [hasLaunchedRecovery]) } diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryDoorOpen.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryDoorOpen.tsx index 17bf1cf0379..3aa1b0f7b74 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryDoorOpen.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryDoorOpen.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { useTranslation } from 'react-i18next' @@ -23,16 +22,33 @@ import { import type { RecoveryContentProps } from './types' +// There are two code paths that render this component: +// 1) The door is open on a route & step in which it is not permitted to have the door open. +// 2) The door is open on a route & step in which it is permitted to have the door open, but the app manually redirects +// to this component. This is commonly done when the route & step itself allows the user to keep the door open, but some +// action on that route & step is about to occur that requires the door to be closed. In this case, once the door event +// has been satisfied, manually route back to the previous route & step. export function RecoveryDoorOpen({ recoveryActionMutationUtils, runStatus, + routeUpdateActions, }: RecoveryContentProps): JSX.Element { const { resumeRecovery, isResumeRecoveryLoading, } = recoveryActionMutationUtils + const { stashedMap, proceedToRouteAndStep } = routeUpdateActions const { t } = useTranslation('error_recovery') + const primaryOnClick = (): void => { + void resumeRecovery().then(() => { + // See comments above for why we do this. + if (stashedMap != null) { + void proceedToRouteAndStep(stashedMap.route, stashedMap.step) + } + }) + } + return ( { - void setRobotInMotion(true) + void handleMotionRouting(true) .then(() => homePipetteZAxes()) - .finally(() => setRobotInMotion(false)) + .finally(() => handleMotionRouting(false)) .then(() => proceedToRouteAndStep(OPTION_SELECTION.ROUTE)) } diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryInProgress.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryInProgress.tsx index 2fba61f0293..4926355113f 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryInProgress.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryInProgress.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/CancelRun.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/CancelRun.tsx index 8f49634a3ad..b3cdd5fe257 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/CancelRun.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/CancelRun.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { useTranslation } from 'react-i18next' import { @@ -23,11 +23,7 @@ import { import { SelectRecoveryOption } from './SelectRecoveryOption' import type { RecoveryContentProps } from '../types' -import type { - RecoveryTipStatusUtils, - UseRecoveryCommandsResult, - UseRouteUpdateActionsResult, -} from '../hooks' +import type { ERUtilsResults } from '../hooks' export function CancelRun(props: RecoveryContentProps): JSX.Element { const { recoveryMap } = props @@ -104,9 +100,9 @@ function CancelRunConfirmation({ } interface OnCancelRunProps { - tipStatusUtils: RecoveryTipStatusUtils - recoveryCommands: UseRecoveryCommandsResult - routeUpdateActions: UseRouteUpdateActionsResult + tipStatusUtils: ERUtilsResults['tipStatusUtils'] + recoveryCommands: ERUtilsResults['recoveryCommands'] + routeUpdateActions: ERUtilsResults['routeUpdateActions'] } // Manages routing to cancel route or drop tip route, depending on tip attachment status. @@ -122,20 +118,20 @@ export function useOnCancelRun({ } { const { ROBOT_CANCELING, DROP_TIP_FLOWS } = RECOVERY_MAP const { isLoadingTipStatus, areTipsAttached } = tipStatusUtils - const { setRobotInMotion, proceedToRouteAndStep } = routeUpdateActions + const { handleMotionRouting, proceedToRouteAndStep } = routeUpdateActions const { cancelRun } = recoveryCommands - const [hasUserClicked, setHasUserClicked] = React.useState(false) + const [hasUserClicked, setHasUserClicked] = useState(false) const showBtnLoadingState = hasUserClicked && isLoadingTipStatus - React.useEffect(() => { + useEffect(() => { if (hasUserClicked) { if (!isLoadingTipStatus) { if (areTipsAttached) { void proceedToRouteAndStep(DROP_TIP_FLOWS.ROUTE) } else { - void setRobotInMotion(true, ROBOT_CANCELING.ROUTE).then(() => { + void handleMotionRouting(true, ROBOT_CANCELING.ROUTE).then(() => { cancelRun() }) } diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/FillWellAndSkip.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/FillWellAndSkip.tsx index 36cce4fdbf9..4fb2290d3ad 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/FillWellAndSkip.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/FillWellAndSkip.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { @@ -81,7 +80,7 @@ export function SkipToNextStep( currentRecoveryOptionUtils, } = props const { - setRobotInMotion, + handleMotionRouting, goBackPrevStep, proceedToRouteAndStep, } = routeUpdateActions @@ -100,7 +99,7 @@ export function SkipToNextStep( } const primaryBtnOnClick = (): Promise => { - return setRobotInMotion(true, ROBOT_SKIPPING_STEP.ROUTE).then(() => { + return handleMotionRouting(true, ROBOT_SKIPPING_STEP.ROUTE).then(() => { skipFailedCommand() }) } diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx index b830f9b3114..7cd9aee55c9 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { @@ -66,7 +65,7 @@ export function BeginRemoval({ const { aPipetteWithTip } = tipStatusUtils const { proceedNextStep, - setRobotInMotion, + handleMotionRouting, proceedToRouteAndStep, } = routeUpdateActions const { cancelRun } = recoveryCommands @@ -85,7 +84,7 @@ export function BeginRemoval({ RETRY_NEW_TIPS.STEPS.REPLACE_TIPS ) } else { - void setRobotInMotion(true, ROBOT_CANCELING.ROUTE).then(() => { + void handleMotionRouting(true, ROBOT_CANCELING.ROUTE).then(() => { cancelRun() }) } @@ -153,7 +152,7 @@ function DropTipFlowsContainer( currentRecoveryOptionUtils, } = props const { DROP_TIP_FLOWS, ROBOT_CANCELING, RETRY_NEW_TIPS } = RECOVERY_MAP - const { proceedToRouteAndStep, setRobotInMotion } = routeUpdateActions + const { proceedToRouteAndStep, handleMotionRouting } = routeUpdateActions const { selectedRecoveryOption } = currentRecoveryOptionUtils const { setTipStatusResolved } = tipStatusUtils const { cancelRun } = recoveryCommands @@ -172,7 +171,7 @@ function DropTipFlowsContainer( } const onEmptyCache = (): void => { - void setRobotInMotion(true, ROBOT_CANCELING.ROUTE).then(() => { + void handleMotionRouting(true, ROBOT_CANCELING.ROUTE).then(() => { cancelRun() }) } diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/RetryNewTips.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/RetryNewTips.tsx index 0a87c7c9234..ee398ac68e3 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/RetryNewTips.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/RetryNewTips.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { LegacyStyledText } from '@opentrons/components' @@ -47,12 +46,12 @@ export function RetryNewTips(props: RecoveryContentProps): JSX.Element { export function RetryWithNewTips(props: RecoveryContentProps): JSX.Element { const { recoveryCommands, routeUpdateActions } = props const { retryFailedCommand, resumeRun } = recoveryCommands - const { setRobotInMotion } = routeUpdateActions + const { handleMotionRouting } = routeUpdateActions const { ROBOT_RETRYING_STEP } = RECOVERY_MAP const { t } = useTranslation('error_recovery') const primaryBtnOnClick = (): Promise => { - return setRobotInMotion(true, ROBOT_RETRYING_STEP.ROUTE) + return handleMotionRouting(true, ROBOT_RETRYING_STEP.ROUTE) .then(() => retryFailedCommand()) .then(() => { resumeRun() diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/RetrySameTips.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/RetrySameTips.tsx index 1d8fa933c42..93a0d84689d 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/RetrySameTips.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/RetrySameTips.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { LegacyStyledText } from '@opentrons/components' @@ -30,12 +29,12 @@ export function RetrySameTips(props: RecoveryContentProps): JSX.Element { export function RetrySameTipsInfo(props: RecoveryContentProps): JSX.Element { const { routeUpdateActions, recoveryCommands } = props const { retryFailedCommand, resumeRun } = recoveryCommands - const { setRobotInMotion } = routeUpdateActions + const { handleMotionRouting } = routeUpdateActions const { ROBOT_RETRYING_STEP } = RECOVERY_MAP const { t } = useTranslation('error_recovery') const primaryBtnOnClick = (): Promise => { - return setRobotInMotion(true, ROBOT_RETRYING_STEP.ROUTE) + return handleMotionRouting(true, ROBOT_RETRYING_STEP.ROUTE) .then(() => retryFailedCommand()) .then(() => { resumeRun() diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/RetryStep.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/RetryStep.tsx index ddaf6b5ee3c..85d6324c6e6 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/RetryStep.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/RetryStep.tsx @@ -1,9 +1,8 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { LegacyStyledText } from '@opentrons/components' -import { RECOVERY_MAP } from '../constants' +import { ERROR_KINDS, RECOVERY_MAP } from '../constants' import { TwoColTextAndFailedStepNextStep } from '../shared' import { SelectRecoveryOption } from './SelectRecoveryOption' @@ -12,11 +11,11 @@ import type { RecoveryContentProps } from '../types' export function RetryStep(props: RecoveryContentProps): JSX.Element { const { recoveryMap } = props const { step, route } = recoveryMap - const { RETRY_FAILED_COMMAND } = RECOVERY_MAP + const { RETRY_STEP } = RECOVERY_MAP const buildContent = (): JSX.Element => { switch (step) { - case RETRY_FAILED_COMMAND.STEPS.CONFIRM_RETRY: + case RETRY_STEP.STEPS.CONFIRM_RETRY: return default: console.warn(`${step} in ${route} not explicitly handled. Rerouting.`) @@ -28,26 +27,34 @@ export function RetryStep(props: RecoveryContentProps): JSX.Element { } export function RetryStepInfo(props: RecoveryContentProps): JSX.Element { - const { routeUpdateActions, recoveryCommands } = props + const { routeUpdateActions, recoveryCommands, errorKind } = props const { ROBOT_RETRYING_STEP } = RECOVERY_MAP const { t } = useTranslation('error_recovery') const { retryFailedCommand, resumeRun } = recoveryCommands - const { setRobotInMotion } = routeUpdateActions + const { handleMotionRouting } = routeUpdateActions const primaryBtnOnClick = (): Promise => { - return setRobotInMotion(true, ROBOT_RETRYING_STEP.ROUTE) + return handleMotionRouting(true, ROBOT_RETRYING_STEP.ROUTE) .then(() => retryFailedCommand()) .then(() => { resumeRun() }) } + const buildBodyCopyKey = (): string => { + switch (errorKind) { + case ERROR_KINDS.TIP_NOT_DETECTED: + return 'take_necessary_actions_failed_pickup' + default: + return 'take_necessary_actions' + } + } const buildBodyText = (): JSX.Element => { return ( , }} diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SelectRecoveryOption.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SelectRecoveryOption.tsx index c56a7ede638..09fa11e8071 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SelectRecoveryOption.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SelectRecoveryOption.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import head from 'lodash/head' import { useTranslation } from 'react-i18next' @@ -59,7 +59,7 @@ export function SelectRecoveryOptionHome({ const { determineTipStatus } = tipStatusUtils const { setSelectedRecoveryOption } = currentRecoveryOptionUtils const validRecoveryOptions = getRecoveryOptions(errorKind) - const [selectedRoute, setSelectedRoute] = React.useState( + const [selectedRoute, setSelectedRoute] = useState( head(validRecoveryOptions) as RecoveryRoute ) @@ -178,7 +178,7 @@ export function DesktopRecoveryOptions({ export function useCurrentTipStatus( determineTipStatus: () => Promise ): void { - React.useEffect(() => { + useEffect(() => { void determineTipStatus() }, []) } @@ -193,6 +193,8 @@ export function getRecoveryOptions(errorKind: ErrorKind): RecoveryRoute[] { return OVERPRESSURE_WHILE_ASPIRATING_OPTIONS case ERROR_KINDS.OVERPRESSURE_WHILE_DISPENSING: return OVERPRESSURE_WHILE_DISPENSING_OPTIONS + case ERROR_KINDS.TIP_NOT_DETECTED: + return TIP_NOT_DETECTED_OPTIONS case ERROR_KINDS.GENERAL_ERROR: return GENERAL_ERROR_OPTIONS } @@ -221,8 +223,14 @@ export const OVERPRESSURE_WHILE_DISPENSING_OPTIONS: RecoveryRoute[] = [ RECOVERY_MAP.CANCEL_RUN.ROUTE, ] +export const TIP_NOT_DETECTED_OPTIONS: RecoveryRoute[] = [ + RECOVERY_MAP.RETRY_STEP.ROUTE, + RECOVERY_MAP.IGNORE_AND_SKIP.ROUTE, + RECOVERY_MAP.CANCEL_RUN.ROUTE, +] + export const GENERAL_ERROR_OPTIONS: RecoveryRoute[] = [ - RECOVERY_MAP.RETRY_FAILED_COMMAND.ROUTE, + RECOVERY_MAP.RETRY_STEP.ROUTE, RECOVERY_MAP.CANCEL_RUN.ROUTE, ] diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SkipStepNewTips.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SkipStepNewTips.tsx index 33c0f199cd8..35d6804cab2 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SkipStepNewTips.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SkipStepNewTips.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { LegacyStyledText } from '@opentrons/components' @@ -49,12 +48,12 @@ export function SkipStepNewTips( export function SkipStepWithNewTips(props: RecoveryContentProps): JSX.Element { const { recoveryCommands, routeUpdateActions } = props const { skipFailedCommand } = recoveryCommands - const { setRobotInMotion } = routeUpdateActions + const { handleMotionRouting } = routeUpdateActions const { ROBOT_SKIPPING_STEP } = RECOVERY_MAP const { t } = useTranslation('error_recovery') const primaryBtnOnClick = (): Promise => { - return setRobotInMotion(true, ROBOT_SKIPPING_STEP.ROUTE).then(() => { + return handleMotionRouting(true, ROBOT_SKIPPING_STEP.ROUTE).then(() => { skipFailedCommand() }) } diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SkipStepSameTips.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SkipStepSameTips.tsx index aed84372ccf..5dddf70970d 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SkipStepSameTips.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SkipStepSameTips.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { LegacyStyledText } from '@opentrons/components' @@ -30,12 +29,12 @@ export function SkipStepSameTips(props: RecoveryContentProps): JSX.Element { export function SkipStepSameTipsInfo(props: RecoveryContentProps): JSX.Element { const { routeUpdateActions, recoveryCommands } = props const { skipFailedCommand } = recoveryCommands - const { setRobotInMotion } = routeUpdateActions + const { handleMotionRouting } = routeUpdateActions const { ROBOT_SKIPPING_STEP } = RECOVERY_MAP const { t } = useTranslation('error_recovery') const primaryBtnOnClick = (): Promise => { - return setRobotInMotion(true, ROBOT_SKIPPING_STEP.ROUTE).then(() => { + return handleMotionRouting(true, ROBOT_SKIPPING_STEP.ROUTE).then(() => { skipFailedCommand() }) } diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/CancelRun.test.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/CancelRun.test.tsx index 3786e50e699..cbf8126e353 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/CancelRun.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/CancelRun.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, describe, it, expect, beforeEach } from 'vitest' import { screen, waitFor } from '@testing-library/react' @@ -23,16 +23,16 @@ describe('RecoveryFooterButtons', () => { const { CANCEL_RUN, ROBOT_CANCELING, DROP_TIP_FLOWS } = RECOVERY_MAP let props: React.ComponentProps let mockGoBackPrevStep: Mock - let mockSetRobotInMotion: Mock + let mockhandleMotionRouting: Mock let mockProceedToRouteAndStep: Mock beforeEach(() => { mockGoBackPrevStep = vi.fn() - mockSetRobotInMotion = vi.fn(() => Promise.resolve()) + mockhandleMotionRouting = vi.fn(() => Promise.resolve()) mockProceedToRouteAndStep = vi.fn() const mockRouteUpdateActions = { goBackPrevStep: mockGoBackPrevStep, - setRobotInMotion: mockSetRobotInMotion, + handleMotionRouting: mockhandleMotionRouting, proceedToRouteAndStep: mockProceedToRouteAndStep, } as any @@ -56,7 +56,10 @@ describe('RecoveryFooterButtons', () => { }) it('renders SelectRecoveryOption when the route is unknown', () => { - props = { ...props, recoveryMap: { ...props.recoveryMap, step: 'UNKNOWN' } } + props = { + ...props, + recoveryMap: { ...props.recoveryMap, step: 'UNKNOWN' as any }, + } render(props) screen.getByText('MOCK SELECT RECOVERY OPTION') @@ -76,7 +79,7 @@ describe('RecoveryFooterButtons', () => { }) it('should call commands in the correct order for the primaryOnClick callback', async () => { - const setRobotInMotionMock = vi.fn(() => Promise.resolve()) + const handleMotionRoutingMock = vi.fn(() => Promise.resolve()) const cancelRunMock = vi.fn(() => Promise.resolve()) const mockRecoveryCommands = { @@ -84,7 +87,7 @@ describe('RecoveryFooterButtons', () => { } as any const mockRouteUpdateActions = { - setRobotInMotion: setRobotInMotionMock, + handleMotionRouting: handleMotionRoutingMock, } as any render({ @@ -96,10 +99,10 @@ describe('RecoveryFooterButtons', () => { clickButtonLabeled('Confirm') await waitFor(() => { - expect(setRobotInMotionMock).toHaveBeenCalledTimes(1) + expect(handleMotionRoutingMock).toHaveBeenCalledTimes(1) }) await waitFor(() => { - expect(setRobotInMotionMock).toHaveBeenCalledWith( + expect(handleMotionRoutingMock).toHaveBeenCalledWith( true, ROBOT_CANCELING.ROUTE ) @@ -108,7 +111,7 @@ describe('RecoveryFooterButtons', () => { expect(cancelRunMock).toHaveBeenCalledTimes(1) }) - expect(setRobotInMotionMock.mock.invocationCallOrder[0]).toBeLessThan( + expect(handleMotionRoutingMock.mock.invocationCallOrder[0]).toBeLessThan( cancelRunMock.mock.invocationCallOrder[0] ) }) @@ -141,7 +144,7 @@ describe('RecoveryFooterButtons', () => { clickButtonLabeled('Confirm') expect(mockProceedToRouteAndStep).not.toHaveBeenCalled() - expect(mockSetRobotInMotion).not.toHaveBeenCalled() + expect(mockhandleMotionRouting).not.toHaveBeenCalled() }) it('should will cancel the run if no tips are detected', () => { @@ -157,6 +160,6 @@ describe('RecoveryFooterButtons', () => { clickButtonLabeled('Confirm') expect(mockProceedToRouteAndStep).not.toHaveBeenCalled() - expect(mockSetRobotInMotion).toHaveBeenCalled() + expect(mockhandleMotionRouting).toHaveBeenCalled() }) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/FillWellAndSkip.test.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/FillWellAndSkip.test.tsx index 4d839340805..ff8d017d7f3 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/FillWellAndSkip.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/FillWellAndSkip.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, expect, beforeEach } from 'vitest' import { screen, waitFor } from '@testing-library/react' @@ -109,7 +109,7 @@ describe('FillWellAndSkip', () => { ...props, recoveryMap: { ...props.recoveryMap, - step: 'UNKNOWN_STEP', + step: 'UNKNOWN_STEP' as any, }, } render(props) @@ -137,13 +137,13 @@ describe('FillWell', () => { describe('SkipToNextStep', () => { let props: React.ComponentProps - let mockSetRobotInMotion: Mock + let mockhandleMotionRouting: Mock let mockGoBackPrevStep: Mock let mockProceedToRouteAndStep: Mock let mockSkipFailedCommand: Mock beforeEach(() => { - mockSetRobotInMotion = vi.fn(() => Promise.resolve()) + mockhandleMotionRouting = vi.fn(() => Promise.resolve()) mockGoBackPrevStep = vi.fn() mockProceedToRouteAndStep = vi.fn() mockSkipFailedCommand = vi.fn(() => Promise.resolve()) @@ -151,7 +151,7 @@ describe('SkipToNextStep', () => { props = { ...mockRecoveryContentProps, routeUpdateActions: { - setRobotInMotion: mockSetRobotInMotion, + handleMotionRouting: mockhandleMotionRouting, goBackPrevStep: mockGoBackPrevStep, proceedToRouteAndStep: mockProceedToRouteAndStep, } as any, @@ -186,7 +186,7 @@ describe('SkipToNextStep', () => { renderSkipToNextStep(props) clickButtonLabeled('Continue run now') await waitFor(() => { - expect(mockSetRobotInMotion).toHaveBeenCalledWith( + expect(mockhandleMotionRouting).toHaveBeenCalledWith( true, RECOVERY_MAP.ROBOT_SKIPPING_STEP.ROUTE ) @@ -195,7 +195,7 @@ describe('SkipToNextStep', () => { expect(mockSkipFailedCommand).toHaveBeenCalled() }) - expect(mockSetRobotInMotion.mock.invocationCallOrder[0]).toBeLessThan( + expect(mockhandleMotionRouting.mock.invocationCallOrder[0]).toBeLessThan( mockSkipFailedCommand.mock.invocationCallOrder[0] ) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/IgnoreErrorSkipStep.test.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/IgnoreErrorSkipStep.test.tsx index 6ddc55506bc..74869c259e5 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/IgnoreErrorSkipStep.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/IgnoreErrorSkipStep.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, expect, beforeEach } from 'vitest' import { screen, fireEvent, waitFor } from '@testing-library/react' @@ -71,7 +71,7 @@ describe('IgnoreErrorSkipStep', () => { ...props, recoveryMap: { ...props.recoveryMap, - step: 'UNKNOWN_STEP', + step: 'UNKNOWN_STEP' as any, }, } render(props) diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/ManageTips.test.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/ManageTips.test.tsx index 373b357d188..f177138f586 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/ManageTips.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/ManageTips.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, expect, beforeEach } from 'vitest' import { screen, @@ -44,12 +44,12 @@ describe('ManageTips', () => { let props: React.ComponentProps let mockProceedNextStep: Mock let mockProceedToRouteAndStep: Mock - let mockSetRobotInMotion: Mock + let mockhandleMotionRouting: Mock beforeEach(() => { mockProceedNextStep = vi.fn() mockProceedToRouteAndStep = vi.fn(() => Promise.resolve()) - mockSetRobotInMotion = vi.fn(() => Promise.resolve()) + mockhandleMotionRouting = vi.fn(() => Promise.resolve()) props = { ...mockRecoveryContentProps, @@ -62,7 +62,7 @@ describe('ManageTips', () => { } as any, routeUpdateActions: { proceedNextStep: mockProceedNextStep, - setRobotInMotion: mockSetRobotInMotion, + handleMotionRouting: mockhandleMotionRouting, proceedToRouteAndStep: mockProceedToRouteAndStep, } as any, recoveryCommands: { cancelRun: vi.fn() } as any, @@ -82,7 +82,10 @@ describe('ManageTips', () => { }) it('renders SelectRecoveryOption when the route is unknown', () => { - props = { ...props, recoveryMap: { ...props.recoveryMap, step: 'UNKNOWN' } } + props = { + ...props, + recoveryMap: { ...props.recoveryMap, step: 'UNKNOWN' as any }, + } render(props) screen.getByText('MOCK SELECT RECOVERY OPTION') @@ -113,7 +116,7 @@ describe('ManageTips', () => { fireEvent.click(skipBtn) clickButtonLabeled('Skip and home pipette') - expect(mockSetRobotInMotion).toHaveBeenCalled() + expect(mockhandleMotionRouting).toHaveBeenCalled() }) it(`handles special routing for ${RETRY_NEW_TIPS.ROUTE} when skipping tip drop`, () => { diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/RetryNewTips.test.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/RetryNewTips.test.tsx index 8ddb269aca5..d7f8469c028 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/RetryNewTips.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/RetryNewTips.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, expect, beforeEach, afterEach } from 'vitest' import { screen, waitFor } from '@testing-library/react' @@ -97,7 +97,7 @@ describe('RetryNewTips', () => { ...props, recoveryMap: { ...props.recoveryMap, - step: 'UNKNOWN_STEP', + step: 'UNKNOWN_STEP' as any, }, } render(props) @@ -122,19 +122,19 @@ describe('RetryNewTips', () => { describe('RetryWithNewTips', () => { let props: React.ComponentProps - let mockSetRobotInMotion: Mock + let mockhandleMotionRouting: Mock let mockRetryFailedCommand: Mock let mockResumeRun: Mock beforeEach(() => { - mockSetRobotInMotion = vi.fn(() => Promise.resolve()) + mockhandleMotionRouting = vi.fn(() => Promise.resolve()) mockRetryFailedCommand = vi.fn(() => Promise.resolve()) mockResumeRun = vi.fn() props = { ...mockRecoveryContentProps, routeUpdateActions: { - setRobotInMotion: mockSetRobotInMotion, + handleMotionRouting: mockhandleMotionRouting, } as any, recoveryCommands: { retryFailedCommand: mockRetryFailedCommand, @@ -159,7 +159,7 @@ describe('RetryWithNewTips', () => { clickButtonLabeled('Retry now') await waitFor(() => { - expect(mockSetRobotInMotion).toHaveBeenCalledWith( + expect(mockhandleMotionRouting).toHaveBeenCalledWith( true, RECOVERY_MAP.ROBOT_RETRYING_STEP.ROUTE ) @@ -171,7 +171,7 @@ describe('RetryWithNewTips', () => { expect(mockResumeRun).toHaveBeenCalled() }) - expect(mockSetRobotInMotion.mock.invocationCallOrder[0]).toBeLessThan( + expect(mockhandleMotionRouting.mock.invocationCallOrder[0]).toBeLessThan( mockRetryFailedCommand.mock.invocationCallOrder[0] ) expect(mockRetryFailedCommand.mock.invocationCallOrder[0]).toBeLessThan( diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/RetrySameTips.test.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/RetrySameTips.test.tsx index b79a96c036d..4514c9cc350 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/RetrySameTips.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/RetrySameTips.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, expect, beforeEach, afterEach } from 'vitest' import { screen, waitFor } from '@testing-library/react' @@ -63,7 +63,7 @@ describe('RetrySameTips', () => { ...props, recoveryMap: { ...props.recoveryMap, - step: 'UNKNOWN_STEP', + step: 'UNKNOWN_STEP' as any, }, } render(props) @@ -73,19 +73,19 @@ describe('RetrySameTips', () => { describe('RetrySameTipsInfo', () => { let props: React.ComponentProps - let mockSetRobotInMotion: Mock + let mockhandleMotionRouting: Mock let mockRetryFailedCommand: Mock let mockResumeRun: Mock beforeEach(() => { - mockSetRobotInMotion = vi.fn(() => Promise.resolve()) + mockhandleMotionRouting = vi.fn(() => Promise.resolve()) mockRetryFailedCommand = vi.fn(() => Promise.resolve()) mockResumeRun = vi.fn() props = { ...mockRecoveryContentProps, routeUpdateActions: { - setRobotInMotion: mockSetRobotInMotion, + handleMotionRouting: mockhandleMotionRouting, } as any, recoveryCommands: { retryFailedCommand: mockRetryFailedCommand, @@ -111,7 +111,7 @@ describe('RetrySameTipsInfo', () => { clickButtonLabeled('Retry now') await waitFor(() => { - expect(mockSetRobotInMotion).toHaveBeenCalledWith( + expect(mockhandleMotionRouting).toHaveBeenCalledWith( true, RECOVERY_MAP.ROBOT_RETRYING_STEP.ROUTE ) @@ -123,7 +123,7 @@ describe('RetrySameTipsInfo', () => { expect(mockResumeRun).toHaveBeenCalled() }) - expect(mockSetRobotInMotion.mock.invocationCallOrder[0]).toBeLessThan( + expect(mockhandleMotionRouting.mock.invocationCallOrder[0]).toBeLessThan( mockRetryFailedCommand.mock.invocationCallOrder[0] ) expect(mockRetryFailedCommand.mock.invocationCallOrder[0]).toBeLessThan( diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/RetryStep.test.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/RetryStep.test.tsx index 25a9deee482..3f0c8179bab 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/RetryStep.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/RetryStep.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, expect, beforeEach, afterEach } from 'vitest' import { screen, waitFor } from '@testing-library/react' @@ -6,7 +6,7 @@ import { mockRecoveryContentProps } from '../../__fixtures__' import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' import { RetryStep, RetryStepInfo } from '../RetryStep' -import { RECOVERY_MAP } from '../../constants' +import { ERROR_KINDS, RECOVERY_MAP } from '../../constants' import { SelectRecoveryOption } from '../SelectRecoveryOption' import { clickButtonLabeled } from '../../__tests__/util' @@ -47,12 +47,12 @@ describe('RetryStep', () => { vi.resetAllMocks() }) - it(`renders RetryStepInfo when step is ${RECOVERY_MAP.RETRY_FAILED_COMMAND.STEPS.CONFIRM_RETRY}`, () => { + it(`renders RetryStepInfo when step is ${RECOVERY_MAP.RETRY_STEP.STEPS.CONFIRM_RETRY}`, () => { props = { ...props, recoveryMap: { ...props.recoveryMap, - step: RECOVERY_MAP.RETRY_FAILED_COMMAND.STEPS.CONFIRM_RETRY, + step: RECOVERY_MAP.RETRY_STEP.STEPS.CONFIRM_RETRY, }, } render(props) @@ -64,7 +64,7 @@ describe('RetryStep', () => { ...props, recoveryMap: { ...props.recoveryMap, - step: 'UNKNOWN_STEP', + step: 'UNKNOWN_STEP' as any, }, } render(props) @@ -74,19 +74,19 @@ describe('RetryStep', () => { describe('RetryStepInfo', () => { let props: React.ComponentProps - let mockSetRobotInMotion: Mock + let mockhandleMotionRouting: Mock let mockRetryFailedCommand: Mock let mockResumeRun: Mock beforeEach(() => { - mockSetRobotInMotion = vi.fn(() => Promise.resolve()) + mockhandleMotionRouting = vi.fn(() => Promise.resolve()) mockRetryFailedCommand = vi.fn(() => Promise.resolve()) mockResumeRun = vi.fn() props = { ...mockRecoveryContentProps, routeUpdateActions: { - setRobotInMotion: mockSetRobotInMotion, + handleMotionRouting: mockhandleMotionRouting, } as any, recoveryCommands: { retryFailedCommand: mockRetryFailedCommand, @@ -99,7 +99,16 @@ describe('RetryStepInfo', () => { vi.resetAllMocks() }) - it('renders the component with the correct text', () => { + it(`renders the component with the correct text for ${ERROR_KINDS.TIP_NOT_DETECTED} `, () => { + renderRetryStepInfo({ ...props, errorKind: ERROR_KINDS.TIP_NOT_DETECTED }) + screen.getByText('Retry step') + screen.queryByText( + 'First, take any necessary actions to prepare the robot to retry the failed tip pickup.' + ) + screen.queryByText('Then, close the robot door before proceeding.') + }) + + it('renders the component with the correct text for not specifically handled error kinds', () => { renderRetryStepInfo(props) screen.getByText('Retry step') screen.queryByText( @@ -113,7 +122,7 @@ describe('RetryStepInfo', () => { clickButtonLabeled('Retry now') await waitFor(() => { - expect(mockSetRobotInMotion).toHaveBeenCalledWith( + expect(mockhandleMotionRouting).toHaveBeenCalledWith( true, RECOVERY_MAP.ROBOT_RETRYING_STEP.ROUTE ) @@ -125,7 +134,7 @@ describe('RetryStepInfo', () => { expect(mockResumeRun).toHaveBeenCalled() }) - expect(mockSetRobotInMotion.mock.invocationCallOrder[0]).toBeLessThan( + expect(mockhandleMotionRouting.mock.invocationCallOrder[0]).toBeLessThan( mockRetryFailedCommand.mock.invocationCallOrder[0] ) expect(mockRetryFailedCommand.mock.invocationCallOrder[0]).toBeLessThan( diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/SelectRecoveryOptions.test.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/SelectRecoveryOptions.test.tsx index 7f4e8932070..fec35b9bc92 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/SelectRecoveryOptions.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/SelectRecoveryOptions.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, describe, it, expect, beforeEach } from 'vitest' import { screen, fireEvent } from '@testing-library/react' import { when } from 'vitest-when' @@ -16,6 +16,7 @@ import { OVERPRESSURE_PREPARE_TO_ASPIRATE, OVERPRESSURE_WHILE_DISPENSING_OPTIONS, NO_LIQUID_DETECTED_OPTIONS, + TIP_NOT_DETECTED_OPTIONS, } from '../SelectRecoveryOption' import { RECOVERY_MAP, ERROR_KINDS } from '../../constants' import { clickButtonLabeled } from '../../__tests__/util' @@ -49,7 +50,7 @@ const renderDesktopRecoveryOptions = ( } describe('SelectRecoveryOption', () => { - const { RETRY_FAILED_COMMAND, RETRY_NEW_TIPS } = RECOVERY_MAP + const { RETRY_STEP, RETRY_NEW_TIPS } = RECOVERY_MAP let props: React.ComponentProps let mockProceedToRouteAndStep: Mock let mockSetSelectedRecoveryOption: Mock @@ -67,8 +68,8 @@ describe('SelectRecoveryOption', () => { ...mockRecoveryContentProps, routeUpdateActions: mockRouteUpdateActions, recoveryMap: { - route: RETRY_FAILED_COMMAND.ROUTE, - step: RETRY_FAILED_COMMAND.STEPS.CONFIRM_RETRY, + route: RETRY_STEP.ROUTE, + step: RETRY_STEP.STEPS.CONFIRM_RETRY, }, tipStatusUtils: { determineTipStatus: vi.fn() } as any, currentRecoveryOptionUtils: { @@ -78,7 +79,7 @@ describe('SelectRecoveryOption', () => { } when(mockGetRecoveryOptionCopy) - .calledWith(RECOVERY_MAP.RETRY_FAILED_COMMAND.ROUTE) + .calledWith(RECOVERY_MAP.RETRY_STEP.ROUTE) .thenReturn('Retry step') when(mockGetRecoveryOptionCopy) .calledWith(RECOVERY_MAP.CANCEL_RUN.ROUTE) @@ -102,9 +103,7 @@ describe('SelectRecoveryOption', () => { clickButtonLabeled('Continue') - expect(mockSetSelectedRecoveryOption).toHaveBeenCalledWith( - RETRY_FAILED_COMMAND.ROUTE - ) + expect(mockSetSelectedRecoveryOption).toHaveBeenCalledWith(RETRY_STEP.ROUTE) }) it('renders appropriate "General Error" copy and click behavior', () => { @@ -121,9 +120,7 @@ describe('SelectRecoveryOption', () => { fireEvent.click(retryStepOption[0]) clickButtonLabeled('Continue') - expect(mockProceedToRouteAndStep).toHaveBeenCalledWith( - RETRY_FAILED_COMMAND.ROUTE - ) + expect(mockProceedToRouteAndStep).toHaveBeenCalledWith(RETRY_STEP.ROUTE) }) it('renders appropriate "Overpressure while aspirating" copy and click behavior', () => { @@ -238,7 +235,7 @@ describe('SelectRecoveryOption', () => { } when(mockGetRecoveryOptionCopy) - .calledWith(RECOVERY_MAP.RETRY_FAILED_COMMAND.ROUTE) + .calledWith(RECOVERY_MAP.RETRY_STEP.ROUTE) .thenReturn('Retry step') when(mockGetRecoveryOptionCopy) .calledWith(RECOVERY_MAP.CANCEL_RUN.ROUTE) @@ -372,4 +369,11 @@ describe('getRecoveryOptions', () => { OVERPRESSURE_WHILE_DISPENSING_OPTIONS ) }) + + it(`returns valid options when the errorKind is ${ERROR_KINDS.TIP_NOT_DETECTED}`, () => { + const overpressureWhileDispensingOptions = getRecoveryOptions( + ERROR_KINDS.TIP_NOT_DETECTED + ) + expect(overpressureWhileDispensingOptions).toBe(TIP_NOT_DETECTED_OPTIONS) + }) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/SkipStepNewTips.test.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/SkipStepNewTips.test.tsx index 389ebe3b7db..73d098f5234 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/SkipStepNewTips.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/SkipStepNewTips.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, expect, beforeEach, afterEach } from 'vitest' import { screen, waitFor } from '@testing-library/react' @@ -97,7 +97,7 @@ describe('SkipStepNewTips', () => { ...props, recoveryMap: { ...props.recoveryMap, - step: 'UNKNOWN_STEP', + step: 'UNKNOWN_STEP' as any, }, } render(props) @@ -122,17 +122,17 @@ describe('SkipStepNewTips', () => { describe('SkipStepWithNewTips', () => { let props: React.ComponentProps - let mockSetRobotInMotion: Mock + let mockhandleMotionRouting: Mock let mockSkipFailedCommand: Mock beforeEach(() => { - mockSetRobotInMotion = vi.fn(() => Promise.resolve()) + mockhandleMotionRouting = vi.fn(() => Promise.resolve()) mockSkipFailedCommand = vi.fn(() => Promise.resolve()) props = { ...mockRecoveryContentProps, routeUpdateActions: { - setRobotInMotion: mockSetRobotInMotion, + handleMotionRouting: mockhandleMotionRouting, } as any, recoveryCommands: { skipFailedCommand: mockSkipFailedCommand, @@ -158,7 +158,7 @@ describe('SkipStepWithNewTips', () => { clickButtonLabeled('Continue run now') await waitFor(() => { - expect(mockSetRobotInMotion).toHaveBeenCalledWith( + expect(mockhandleMotionRouting).toHaveBeenCalledWith( true, RECOVERY_MAP.ROBOT_SKIPPING_STEP.ROUTE ) @@ -167,7 +167,7 @@ describe('SkipStepWithNewTips', () => { expect(mockSkipFailedCommand).toHaveBeenCalled() }) - expect(mockSetRobotInMotion.mock.invocationCallOrder[0]).toBeLessThan( + expect(mockhandleMotionRouting.mock.invocationCallOrder[0]).toBeLessThan( mockSkipFailedCommand.mock.invocationCallOrder[0] ) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/SkipStepSameTips.test.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/SkipStepSameTips.test.tsx index c17957103f6..1a9b7b5dcd2 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/SkipStepSameTips.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/SkipStepSameTips.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, expect, beforeEach, afterEach } from 'vitest' import { screen, waitFor } from '@testing-library/react' @@ -60,7 +60,7 @@ describe('SkipStepSameTips', () => { ...props, recoveryMap: { ...props.recoveryMap, - step: 'UNKNOWN_STEP', + step: 'UNKNOWN_STEP' as any, }, } render(props) @@ -70,17 +70,17 @@ describe('SkipStepSameTips', () => { describe('SkipStepSameTipsInfo', () => { let props: React.ComponentProps - let mockSetRobotInMotion: Mock + let mockhandleMotionRouting: Mock let mockSkipFailedCommand: Mock beforeEach(() => { - mockSetRobotInMotion = vi.fn(() => Promise.resolve()) + mockhandleMotionRouting = vi.fn(() => Promise.resolve()) mockSkipFailedCommand = vi.fn(() => Promise.resolve()) props = { ...mockRecoveryContentProps, routeUpdateActions: { - setRobotInMotion: mockSetRobotInMotion, + handleMotionRouting: mockhandleMotionRouting, } as any, recoveryCommands: { skipFailedCommand: mockSkipFailedCommand, @@ -106,7 +106,7 @@ describe('SkipStepSameTipsInfo', () => { clickButtonLabeled('Continue run now') await waitFor(() => { - expect(mockSetRobotInMotion).toHaveBeenCalledWith( + expect(mockhandleMotionRouting).toHaveBeenCalledWith( true, RECOVERY_MAP.ROBOT_SKIPPING_STEP.ROUTE ) @@ -116,7 +116,7 @@ describe('SkipStepSameTipsInfo', () => { expect(mockSkipFailedCommand).toHaveBeenCalled() }) - expect(mockSetRobotInMotion.mock.invocationCallOrder[0]).toBeLessThan( + expect(mockhandleMotionRouting.mock.invocationCallOrder[0]).toBeLessThan( mockSkipFailedCommand.mock.invocationCallOrder[0] ) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/RunPausedSplash.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoverySplash.tsx similarity index 62% rename from app/src/organisms/ErrorRecoveryFlows/RunPausedSplash.tsx rename to app/src/organisms/ErrorRecoveryFlows/RecoverySplash.tsx index 4e2642aac85..f178240c6ef 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RunPausedSplash.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoverySplash.tsx @@ -1,36 +1,41 @@ -import * as React from 'react' +import { useEffect } from 'react' import styled, { css } from 'styled-components' import { useTranslation } from 'react-i18next' import { - Flex, - Icon, - JUSTIFY_CENTER, ALIGN_CENTER, - SPACING, COLORS, DIRECTION_COLUMN, - POSITION_ABSOLUTE, - TYPOGRAPHY, - OVERFLOW_WRAP_BREAK_WORD, DISPLAY_FLEX, + Flex, + Icon, + JUSTIFY_CENTER, JUSTIFY_SPACE_BETWEEN, - TEXT_ALIGN_CENTER, - StyledText, + OVERFLOW_WRAP_BREAK_WORD, + POSITION_ABSOLUTE, PrimaryButton, SecondaryButton, + SPACING, + StyledText, + TEXT_ALIGN_CENTER, + TYPOGRAPHY, LargeButton, + WARNING_TOAST, } from '@opentrons/components' +import { + RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR, + RUN_STATUS_AWAITING_RECOVERY_PAUSED, +} from '@opentrons/api-client' import { useErrorName } from './hooks' import { getErrorKind } from './utils' - import { BANNER_TEXT_CONTAINER_STYLE, BANNER_TEXT_CONTENT_STYLE, RECOVERY_MAP, } from './constants' import { RecoveryInterventionModal, StepInfo } from './shared' +import { useToaster } from '../ToasterOven' import type { RobotType } from '@opentrons/shared-data' import type { ErrorRecoveryFlowsProps } from '.' @@ -40,10 +45,9 @@ import type { useRetainedFailedCommandBySource, } from './hooks' import type { RecoveryRoute, RouteStep } from './types' - import type { UseRecoveryAnalyticsResult } from '/app/redux-resources/analytics' -export function useRunPausedSplash( +export function useRecoverySplash( isOnDevice: boolean, showERWizard: boolean ): boolean { @@ -56,18 +60,18 @@ export function useRunPausedSplash( } } -type RunPausedSplashProps = ERUtilsResults & { - isOnDevice: boolean - failedCommand: ReturnType - protocolAnalysis: ErrorRecoveryFlowsProps['protocolAnalysis'] - robotType: RobotType - robotName: string - toggleERWizAsActiveUser: UseRecoveryTakeoverResult['toggleERWizAsActiveUser'] - analytics: UseRecoveryAnalyticsResult -} -export function RunPausedSplash( - props: RunPausedSplashProps -): JSX.Element | null { +type RecoverySplashProps = ErrorRecoveryFlowsProps & + ERUtilsResults & { + isOnDevice: boolean + failedCommand: ReturnType + robotType: RobotType + robotName: string + toggleERWizAsActiveUser: UseRecoveryTakeoverResult['toggleERWizAsActiveUser'] + analytics: UseRecoveryAnalyticsResult + /* Whether the app should resume any paused recovery state without user action. */ + resumePausedRecovery: boolean + } +export function RecoverySplash(props: RecoverySplashProps): JSX.Element | null { const { isOnDevice, toggleERWizAsActiveUser, @@ -75,10 +79,14 @@ export function RunPausedSplash( failedCommand, analytics, robotName, + runStatus, + recoveryActionMutationUtils, + resumePausedRecovery, } = props const { t } = useTranslation('error_recovery') const errorKind = getErrorKind(failedCommand?.byRunRecord ?? null) const title = useErrorName(errorKind) + const { makeToast } = useToaster() const { proceedToRouteAndStep } = routeUpdateActions const { reportErrorEvent } = analytics @@ -91,18 +99,58 @@ export function RunPausedSplash( ) } + // Resume recovery when the run when the door is closed. + // The CTA/flow for handling a door open event within the ER wizard is different, and because this splash always renders + // behind the wizard, we want to ensure we only implicitly resume recovery when only viewing the splash from this app. + useEffect(() => { + if ( + runStatus === RUN_STATUS_AWAITING_RECOVERY_PAUSED && + resumePausedRecovery + ) { + recoveryActionMutationUtils.resumeRecovery() + } + }, [runStatus, resumePausedRecovery]) + const buildDoorOpenAlert = (): void => { + makeToast(t('close_door_to_resume') as string, WARNING_TOAST) + } + + const handleConditionalClick = (onClick: () => void): void => { + switch (runStatus) { + case RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR: + buildDoorOpenAlert() + break + default: + onClick() + break + } + } // Do not launch error recovery, but do utilize the wizard's cancel route. - const onCancelClick = (): Promise => { - return toggleERWizAsActiveUser(true, false).then(() => { - reportErrorEvent(failedCommand?.byRunRecord ?? null, 'cancel-run') - void proceedToRouteAndStep(RECOVERY_MAP.CANCEL_RUN.ROUTE) - }) + const onCancelClick = (): void => { + const onClick = (): void => { + void toggleERWizAsActiveUser(true, false).then(() => { + reportErrorEvent(failedCommand?.byRunRecord ?? null, 'cancel-run') + void proceedToRouteAndStep(RECOVERY_MAP.CANCEL_RUN.ROUTE) + }) + } + handleConditionalClick(onClick) + } + + const onLaunchERClick = (): void => { + const onClick = (): void => { + void toggleERWizAsActiveUser(true, true).then(() => { + reportErrorEvent(failedCommand?.byRunRecord ?? null, 'launch-recovery') + }) + } + handleConditionalClick(onClick) } - const onLaunchERClick = (): Promise => { - return toggleERWizAsActiveUser(true, true).then(() => { - reportErrorEvent(failedCommand?.byRunRecord ?? null, 'launch-recovery') - }) + const isDisabled = (): boolean => { + switch (runStatus) { + case RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR: + return true + default: + return false + } } // TODO(jh 05-22-24): The hardcoded Z-indexing is non-ideal but must be done to keep the splash page above @@ -152,14 +200,18 @@ export function RunPausedSplash( @@ -197,12 +249,20 @@ export function RunPausedSplash( - + {t('cancel_run')} {t('launch_recovery_mode')} @@ -237,6 +297,30 @@ const SHARED_BUTTON_STYLE_ODD = css` width: 29rem; height: 13.5rem; ` +const BTN_STYLE_DISABLED_ODD = css` + ${SHARED_BUTTON_STYLE_ODD} + + background-color: ${COLORS.grey35}; + color: ${COLORS.grey50}; + border: none; + box-shadow: none; + + #btn-icon: { + color: ${COLORS.grey50}; + } + + &:active, + &:focus, + &:hover { + background-color: ${COLORS.grey35}; + color: ${COLORS.grey50}; + } + &:active, + &:focus, + &:hover #btn-icon { + color: ${COLORS.grey50}; + } +` const PRIMARY_BTN_STYLES_DESKTOP = css` background-color: ${COLORS.red50}; @@ -248,3 +332,17 @@ const PRIMARY_BTN_STYLES_DESKTOP = css` background-color: ${COLORS.red55}; } ` +const BTN_STYLES_DISABLED_DESKTOP = css` + background-color: ${COLORS.grey30}; + color: ${COLORS.grey40}; + border: none; + box-shadow: none; + + &:active, + &:focus, + &:hover { + background-color: ${COLORS.grey30}; + color: ${COLORS.grey40}; + } + cursor: default; +` diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryTakeover.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryTakeover.tsx index f42201a7f8f..d94732ea129 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryTakeover.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryTakeover.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { @@ -99,7 +99,7 @@ export function RecoveryTakeoverODD({ clearClientData, isRunStatusAwaitingRecovery, }: RecoveryTakeoverProps): JSX.Element { - const [showConfirmation, setShowConfirmation] = React.useState(false) + const [showConfirmation, setShowConfirmation] = useState(false) return ( { @@ -144,17 +143,17 @@ describe('ErrorRecoveryFlows', () => { protocolAnalysis: {} as any, } vi.mocked(ErrorRecoveryWizard).mockReturnValue(
MOCK WIZARD
) - vi.mocked(RunPausedSplash).mockReturnValue( -
MOCK RUN PAUSED SPLASH
- ) + vi.mocked(RecoverySplash).mockReturnValue(
MOCK RUN PAUSED SPLASH
) vi.mocked(useERWizard).mockReturnValue({ hasLaunchedRecovery: true, toggleERWizard: () => Promise.resolve(), showERWizard: true, }) - vi.mocked(useRunPausedSplash).mockReturnValue(true) - vi.mocked(useERUtils).mockReturnValue({ routeUpdateActions: {} } as any) - vi.mocked(useShowDoorInfo).mockReturnValue(false) + vi.mocked(useRecoverySplash).mockReturnValue(true) + vi.mocked(useERUtils).mockReturnValue({ + routeUpdateActions: {}, + doorStatusUtils: { isDoorOpen: false, isProhibitedDoorOpen: false }, + } as any) vi.mocked(useRecoveryAnalytics).mockReturnValue({ reportErrorEvent: vi.fn(), } as any) @@ -174,13 +173,17 @@ describe('ErrorRecoveryFlows', () => { }) it('renders the wizard when isDoorOpen is true', () => { - vi.mocked(useShowDoorInfo).mockReturnValue(true) vi.mocked(useERWizard).mockReturnValue({ hasLaunchedRecovery: false, toggleERWizard: () => Promise.resolve(), showERWizard: false, }) + vi.mocked(useERUtils).mockReturnValue({ + routeUpdateActions: {}, + doorStatusUtils: { isDoorOpen: true, isProhibitedDoorOpen: true }, + } as any) + render(props) screen.getByText('MOCK WIZARD') }) @@ -191,7 +194,10 @@ describe('ErrorRecoveryFlows', () => { toggleERWizard: () => Promise.resolve(), showERWizard: false, }) - vi.mocked(useShowDoorInfo).mockReturnValue(false) + vi.mocked(useERUtils).mockReturnValue({ + routeUpdateActions: {}, + doorStatusUtils: { isDoorOpen: false, isProhibitedDoorOpen: false }, + } as any) render(props) expect(screen.queryByText('MOCK WIZARD')).not.toBeInTheDocument() @@ -203,7 +209,7 @@ describe('ErrorRecoveryFlows', () => { }) it('does not render the splash when showSplash is false', () => { - vi.mocked(useRunPausedSplash).mockReturnValue(false) + vi.mocked(useRecoverySplash).mockReturnValue(false) render(props) expect(screen.queryByText('MOCK RUN PAUSED SPLASH')).not.toBeInTheDocument() }) diff --git a/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryWizard.test.tsx b/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryWizard.test.tsx index 312c90cc7cc..6d35c993e90 100644 --- a/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryWizard.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryWizard.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, describe, it, expect, beforeEach } from 'vitest' import { renderHook, act, screen, waitFor } from '@testing-library/react' @@ -134,7 +134,10 @@ describe('ErrorRecoveryComponent', () => { }) it('renders the recovery door modal when isDoorOpen is true', () => { - props = { ...props, isDoorOpen: true } + props = { + ...props, + doorStatusUtils: { isProhibitedDoorOpen: true, isDoorOpen: true }, + } renderRecoveryComponent(props) @@ -159,7 +162,7 @@ const renderRecoveryContent = ( describe('ErrorRecoveryContent', () => { const { OPTION_SELECTION, - RETRY_FAILED_COMMAND, + RETRY_STEP, ROBOT_CANCELING, ROBOT_RESUMING, ROBOT_IN_MOTION, @@ -175,6 +178,7 @@ describe('ErrorRecoveryContent', () => { CANCEL_RUN, DROP_TIP_FLOWS, ERROR_WHILE_RECOVERING, + ROBOT_DOOR_OPEN, } = RECOVERY_MAP let props: React.ComponentProps @@ -202,6 +206,7 @@ describe('ErrorRecoveryContent', () => { vi.mocked(IgnoreErrorSkipStep).mockReturnValue(
MOCK_IGNORE_ERROR_SKIP_STEP
) + vi.mocked(RecoveryDoorOpen).mockReturnValue(
MOCK_DOOR_OPEN
) }) it(`returns SelectRecoveryOption when the route is ${OPTION_SELECTION.ROUTE}`, () => { @@ -210,12 +215,12 @@ describe('ErrorRecoveryContent', () => { screen.getByText('MOCK_SELECT_RECOVERY_OPTION') }) - it(`returns ResumeRun when the route is ${RETRY_FAILED_COMMAND.ROUTE}`, () => { + it(`returns ResumeRun when the route is ${RETRY_STEP.ROUTE}`, () => { props = { ...props, recoveryMap: { ...props.recoveryMap, - route: RETRY_FAILED_COMMAND.ROUTE, + route: RETRY_STEP.ROUTE, }, } renderRecoveryContent(props) @@ -417,26 +422,39 @@ describe('ErrorRecoveryContent', () => { screen.getByText('MOCK_IN_PROGRESS') }) + + it(`returns RecoveryDoorOpen when the route is ${ROBOT_DOOR_OPEN.ROUTE}`, () => { + props = { + ...props, + recoveryMap: { + ...props.recoveryMap, + route: ROBOT_DOOR_OPEN.ROUTE, + }, + } + renderRecoveryContent(props) + + screen.getByText('MOCK_DOOR_OPEN') + }) }) describe('useInitialPipetteHome', () => { let mockZHomePipetteZAxes: Mock - let mockSetRobotInMotion: Mock + let mockhandleMotionRouting: Mock let mockRecoveryCommands: any let mockRouteUpdateActions: any beforeEach(() => { mockZHomePipetteZAxes = vi.fn() - mockSetRobotInMotion = vi.fn() + mockhandleMotionRouting = vi.fn() - mockSetRobotInMotion.mockResolvedValue(() => mockZHomePipetteZAxes()) - mockZHomePipetteZAxes.mockResolvedValue(() => mockSetRobotInMotion()) + mockhandleMotionRouting.mockResolvedValue(() => mockZHomePipetteZAxes()) + mockZHomePipetteZAxes.mockResolvedValue(() => mockhandleMotionRouting()) mockRecoveryCommands = { homePipetteZAxes: mockZHomePipetteZAxes, } as any mockRouteUpdateActions = { - setRobotInMotion: mockSetRobotInMotion, + handleMotionRouting: mockhandleMotionRouting, } as any }) @@ -449,7 +467,7 @@ describe('useInitialPipetteHome', () => { }) ) - expect(mockSetRobotInMotion).not.toHaveBeenCalled() + expect(mockhandleMotionRouting).not.toHaveBeenCalled() }) it('sets the motion screen properly and z-homes all pipettes only on the initial render of Error Recovery', async () => { @@ -462,26 +480,26 @@ describe('useInitialPipetteHome', () => { ) await waitFor(() => { - expect(mockSetRobotInMotion).toHaveBeenCalledWith(true) + expect(mockhandleMotionRouting).toHaveBeenCalledWith(true) }) await waitFor(() => { expect(mockZHomePipetteZAxes).toHaveBeenCalledTimes(1) }) await waitFor(() => { - expect(mockSetRobotInMotion).toHaveBeenCalledWith(false) + expect(mockhandleMotionRouting).toHaveBeenCalledWith(false) }) - expect(mockSetRobotInMotion.mock.invocationCallOrder[0]).toBeLessThan( + expect(mockhandleMotionRouting.mock.invocationCallOrder[0]).toBeLessThan( mockZHomePipetteZAxes.mock.invocationCallOrder[0] ) expect(mockZHomePipetteZAxes.mock.invocationCallOrder[0]).toBeLessThan( - mockSetRobotInMotion.mock.invocationCallOrder[1] + mockhandleMotionRouting.mock.invocationCallOrder[1] ) rerender() await waitFor(() => { - expect(mockSetRobotInMotion).toHaveBeenCalledTimes(2) + expect(mockhandleMotionRouting).toHaveBeenCalledTimes(2) }) await waitFor(() => { expect(mockZHomePipetteZAxes).toHaveBeenCalledTimes(1) diff --git a/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryDoorOpen.test.tsx b/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryDoorOpen.test.tsx index aed255ba9a1..a9fc92e1b84 100644 --- a/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryDoorOpen.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryDoorOpen.test.tsx @@ -1,5 +1,6 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, beforeEach, vi, expect } from 'vitest' + import { screen } from '@testing-library/react' import { RUN_STATUS_AWAITING_RECOVERY_PAUSED } from '@opentrons/api-client' @@ -7,9 +8,9 @@ import { renderWithProviders } from '/app/__testing-utils__' import { mockRecoveryContentProps } from '../__fixtures__' import { i18n } from '/app/i18n' import { RecoveryDoorOpen } from '../RecoveryDoorOpen' +import { clickButtonLabeled } from './util' import type { Mock } from 'vitest' -import { clickButtonLabeled } from './util' const render = (props: React.ComponentProps) => { return renderWithProviders(, { @@ -20,9 +21,11 @@ const render = (props: React.ComponentProps) => { describe('RecoveryDoorOpen', () => { let props: React.ComponentProps let mockResumeRecovery: Mock + let mockProceedToRouteAndStep: Mock beforeEach(() => { - mockResumeRecovery = vi.fn() + mockResumeRecovery = vi.fn().mockResolvedValue(undefined) + mockProceedToRouteAndStep = vi.fn() props = { ...mockRecoveryContentProps, recoveryActionMutationUtils: { @@ -30,6 +33,10 @@ describe('RecoveryDoorOpen', () => { isResumeRecoveryLoading: false, }, runStatus: RUN_STATUS_AWAITING_RECOVERY_PAUSED, + routeUpdateActions: { + stashedMap: null, + proceedToRouteAndStep: mockProceedToRouteAndStep, + } as any, } }) @@ -50,4 +57,22 @@ describe('RecoveryDoorOpen', () => { expect(mockResumeRecovery).toHaveBeenCalledTimes(1) }) + + it('calls proceedToRouteAndStep after resumeRecovery if stashedMap is provided', async () => { + const stashedMap = { route: 'testRoute', step: 'testStep' } as any + props.routeUpdateActions.stashedMap = stashedMap + + render(props) + + clickButtonLabeled('Resume') + + await vi.waitFor(() => { + expect(mockResumeRecovery).toHaveBeenCalledTimes(1) + expect(mockProceedToRouteAndStep).toHaveBeenCalledTimes(1) + expect(mockProceedToRouteAndStep).toHaveBeenCalledWith( + stashedMap.route, + stashedMap.step + ) + }) + }) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryError.test.tsx b/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryError.test.tsx index 4e2a415d8ea..1394993746b 100644 --- a/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryError.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryError.test.tsx @@ -1,5 +1,5 @@ /* eslint-disable testing-library/prefer-presence-queries */ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, expect, beforeEach } from 'vitest' import { screen, fireEvent, waitFor } from '@testing-library/react' @@ -23,13 +23,13 @@ describe('RecoveryError', () => { let props: React.ComponentProps let proceedToRouteAndStepMock: Mock let getRecoverOptionCopyMock: Mock - let setRobotInMotionMock: Mock + let handleMotionRoutingMock: Mock let homePipetteZAxesMock: Mock beforeEach(() => { proceedToRouteAndStepMock = vi.fn() getRecoverOptionCopyMock = vi.fn() - setRobotInMotionMock = vi.fn().mockResolvedValue(undefined) + handleMotionRoutingMock = vi.fn().mockResolvedValue(undefined) homePipetteZAxesMock = vi.fn().mockResolvedValue(undefined) props = { @@ -37,7 +37,7 @@ describe('RecoveryError', () => { routeUpdateActions: { ...mockRecoveryContentProps.routeUpdateActions, proceedToRouteAndStep: proceedToRouteAndStepMock, - setRobotInMotion: setRobotInMotionMock, + handleMotionRouting: handleMotionRoutingMock, }, recoveryCommands: { ...mockRecoveryContentProps.recoveryCommands, @@ -151,14 +151,14 @@ describe('RecoveryError', () => { fireEvent.click(screen.queryAllByText('Back to menu')[0]) - expect(setRobotInMotionMock).toHaveBeenCalledWith(true) + expect(handleMotionRoutingMock).toHaveBeenCalledWith(true) await waitFor(() => { expect(homePipetteZAxesMock).toHaveBeenCalled() }) await waitFor(() => { - expect(setRobotInMotionMock).toHaveBeenCalledWith(false) + expect(handleMotionRoutingMock).toHaveBeenCalledWith(false) }) await waitFor(() => { @@ -167,17 +167,17 @@ describe('RecoveryError', () => { ) }) - expect(setRobotInMotionMock).toHaveBeenCalledTimes(2) + expect(handleMotionRoutingMock).toHaveBeenCalledTimes(2) expect(homePipetteZAxesMock).toHaveBeenCalledTimes(1) expect(proceedToRouteAndStepMock).toHaveBeenCalledTimes(1) - expect(setRobotInMotionMock.mock.invocationCallOrder[0]).toBeLessThan( + expect(handleMotionRoutingMock.mock.invocationCallOrder[0]).toBeLessThan( homePipetteZAxesMock.mock.invocationCallOrder[0] ) expect(homePipetteZAxesMock.mock.invocationCallOrder[0]).toBeLessThan( - setRobotInMotionMock.mock.invocationCallOrder[1] + handleMotionRoutingMock.mock.invocationCallOrder[1] ) - expect(setRobotInMotionMock.mock.invocationCallOrder[1]).toBeLessThan( + expect(handleMotionRoutingMock.mock.invocationCallOrder[1]).toBeLessThan( proceedToRouteAndStepMock.mock.invocationCallOrder[0] ) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryInProgress.test.tsx b/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryInProgress.test.tsx index 75f96a88dab..20e735e950f 100644 --- a/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryInProgress.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryInProgress.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { beforeEach, describe, it } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/ErrorRecoveryFlows/__tests__/RunPausedSplash.test.tsx b/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoverySplash.test.tsx similarity index 72% rename from app/src/organisms/ErrorRecoveryFlows/__tests__/RunPausedSplash.test.tsx rename to app/src/organisms/ErrorRecoveryFlows/__tests__/RecoverySplash.test.tsx index f1d5e17b616..da3ddc07629 100644 --- a/app/src/organisms/ErrorRecoveryFlows/__tests__/RunPausedSplash.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoverySplash.test.tsx @@ -1,22 +1,31 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' import { fireEvent, screen, waitFor, renderHook } from '@testing-library/react' import { createStore } from 'redux' +import { QueryClient, QueryClientProvider } from 'react-query' +import { Provider } from 'react-redux' + +import { + RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR, + RUN_STATUS_AWAITING_RECOVERY, + RUN_STATUS_AWAITING_RECOVERY_PAUSED, +} from '@opentrons/api-client' import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' import { mockRecoveryContentProps } from '../__fixtures__' import { getIsOnDevice } from '/app/redux/config' -import { useRunPausedSplash, RunPausedSplash } from '../RunPausedSplash' +import { useRecoverySplash, RecoverySplash } from '../RecoverySplash' import { StepInfo } from '../shared' +import { useToaster } from '../../ToasterOven' +import { clickButtonLabeled } from './util' import type { Store } from 'redux' -import { QueryClient, QueryClientProvider } from 'react-query' -import { Provider } from 'react-redux' vi.mock('/app/redux/config') vi.mock('../shared') +vi.mock('../../ToasterOven') const store: Store = createStore(vi.fn(), {}) @@ -45,7 +54,7 @@ describe('useRunPausedSplash', () => { TEST_CASES.forEach(({ isOnDevice, showERWizard, expected }) => { it(`returns ${expected} when isOnDevice is ${isOnDevice} and showERWizard is ${showERWizard}`, () => { const { result } = renderHook( - () => useRunPausedSplash(isOnDevice, showERWizard), + () => useRecoverySplash(isOnDevice, showERWizard), { wrapper, } @@ -56,10 +65,10 @@ describe('useRunPausedSplash', () => { }) }) -const render = (props: React.ComponentProps) => { +const render = (props: React.ComponentProps) => { return renderWithProviders( - + , { i18nInstance: i18n, @@ -67,13 +76,15 @@ const render = (props: React.ComponentProps) => { ) } -describe('RunPausedSplash', () => { - let props: React.ComponentProps +describe('RecoverySplash', () => { + let props: React.ComponentProps const mockToggleERWiz = vi.fn(() => Promise.resolve()) const mockProceedToRouteAndStep = vi.fn() const mockRouteUpdateActions = { proceedToRouteAndStep: mockProceedToRouteAndStep, } as any + const mockMakeToast = vi.fn() + const mockResumeRecovery = vi.fn() beforeEach(() => { props = { @@ -81,9 +92,14 @@ describe('RunPausedSplash', () => { robotName: 'testRobot', toggleERWizAsActiveUser: mockToggleERWiz, routeUpdateActions: mockRouteUpdateActions, + recoveryActionMutationUtils: { + resumeRecovery: mockResumeRecovery, + } as any, + resumePausedRecovery: true, } vi.mocked(StepInfo).mockReturnValue(
MOCK STEP INFO
) + vi.mocked(useToaster).mockReturnValue({ makeToast: mockMakeToast } as any) }) afterEach(() => { @@ -147,4 +163,25 @@ describe('RunPausedSplash', () => { expect(mockToggleERWiz).toHaveBeenCalledWith(true, true) }) }) + + it('should render a door open toast if the door is open', () => { + props = { + ...props, + runStatus: RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR, + } + + render(props) + + clickButtonLabeled('Launch Recovery Mode') + + expect(mockMakeToast).toHaveBeenCalled() + }) + + it(`should transition the run status from ${RUN_STATUS_AWAITING_RECOVERY_PAUSED} to ${RUN_STATUS_AWAITING_RECOVERY} when resumePausedRecovery is true`, () => { + props = { ...props, runStatus: RUN_STATUS_AWAITING_RECOVERY_PAUSED } + + render(props) + + expect(mockResumeRecovery).toHaveBeenCalled() + }) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryTakeover.test.tsx b/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryTakeover.test.tsx index 6a3fe495366..1eec7782713 100644 --- a/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryTakeover.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryTakeover.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, expect, beforeEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/ErrorRecoveryFlows/constants.ts b/app/src/organisms/ErrorRecoveryFlows/constants.ts index faadf0730aa..f7cea32eb03 100644 --- a/app/src/organisms/ErrorRecoveryFlows/constants.ts +++ b/app/src/organisms/ErrorRecoveryFlows/constants.ts @@ -10,13 +10,14 @@ import { TEXT_ALIGN_CENTER, } from '@opentrons/components' -import type { StepOrder } from './types' +import type { RecoveryRouteStepMetadata, StepOrder } from './types' // Server-defined error types. // (Values for the .error.errorType property of a run command.) export const DEFINED_ERROR_TYPES = { OVERPRESSURE: 'overpressure', LIQUID_NOT_FOUND: 'liquidNotFound', + TIP_PHYSICALLY_MISSING: 'tipPhysicallyMissing', } // Client-defined error-handling flows. @@ -26,6 +27,7 @@ export const ERROR_KINDS = { OVERPRESSURE_PREPARE_TO_ASPIRATE: 'OVERPRESSURE_PREPARE_TO_ASPIRATE', OVERPRESSURE_WHILE_ASPIRATING: 'OVERPRESSURE_WHILE_ASPIRATING', OVERPRESSURE_WHILE_DISPENSING: 'OVERPRESSURE_WHILE_DISPENSING', + TIP_NOT_DETECTED: 'TIP_NOT_DETECTED', } as const // TODO(jh, 05-09-24): Refactor to a directed graph. EXEC-430. @@ -86,6 +88,12 @@ export const RECOVERY_MAP = { SKIPPING: 'skipping', }, }, + ROBOT_DOOR_OPEN: { + ROUTE: 'door', + STEPS: { + DOOR_OPEN: 'door-open', + }, + }, // Recovery options below OPTION_SELECTION: { ROUTE: 'option-selection', @@ -104,8 +112,8 @@ export const RECOVERY_MAP = { STEPS: { MANUALLY_FILL: 'manually-fill', SKIP: 'skip' }, }, REFILL_AND_RESUME: { ROUTE: 'refill-and-resume', STEPS: {} }, - RETRY_FAILED_COMMAND: { - ROUTE: 'retry-failed-command', + RETRY_STEP: { + ROUTE: 'retry-step', STEPS: { CONFIRM_RETRY: 'confirm-retry' }, }, RETRY_NEW_TIPS: { @@ -142,13 +150,14 @@ export const RECOVERY_MAP = { const { OPTION_SELECTION, - RETRY_FAILED_COMMAND, + RETRY_STEP, ROBOT_CANCELING, ROBOT_PICKING_UP_TIPS, ROBOT_RESUMING, ROBOT_IN_MOTION, ROBOT_RETRYING_STEP, ROBOT_SKIPPING_STEP, + ROBOT_DOOR_OPEN, DROP_TIP_FLOWS, REFILL_AND_RESUME, IGNORE_AND_SKIP, @@ -164,7 +173,7 @@ const { // The deterministic ordering of steps for a given route. export const STEP_ORDER: StepOrder = { [OPTION_SELECTION.ROUTE]: [OPTION_SELECTION.STEPS.SELECT], - [RETRY_FAILED_COMMAND.ROUTE]: [RETRY_FAILED_COMMAND.STEPS.CONFIRM_RETRY], + [RETRY_STEP.ROUTE]: [RETRY_STEP.STEPS.CONFIRM_RETRY], [RETRY_NEW_TIPS.ROUTE]: [ RETRY_NEW_TIPS.STEPS.DROP_TIPS, RETRY_NEW_TIPS.STEPS.REPLACE_TIPS, @@ -185,6 +194,7 @@ export const STEP_ORDER: StepOrder = { [ROBOT_RESUMING.ROUTE]: [ROBOT_RESUMING.STEPS.RESUMING], [ROBOT_RETRYING_STEP.ROUTE]: [ROBOT_RETRYING_STEP.STEPS.RETRYING], [ROBOT_SKIPPING_STEP.ROUTE]: [ROBOT_SKIPPING_STEP.STEPS.SKIPPING], + [ROBOT_DOOR_OPEN.ROUTE]: [ROBOT_DOOR_OPEN.STEPS.DOOR_OPEN], [DROP_TIP_FLOWS.ROUTE]: [ DROP_TIP_FLOWS.STEPS.BEGIN_REMOVAL, DROP_TIP_FLOWS.STEPS.BEFORE_BEGINNING, @@ -206,6 +216,108 @@ export const STEP_ORDER: StepOrder = { ], } +// Contains metadata specific to all routes and/or steps. +export const RECOVERY_MAP_METADATA: RecoveryRouteStepMetadata = { + [DROP_TIP_FLOWS.ROUTE]: { + [DROP_TIP_FLOWS.STEPS.BEGIN_REMOVAL]: { allowDoorOpen: false }, + [DROP_TIP_FLOWS.STEPS.BEFORE_BEGINNING]: { + allowDoorOpen: false, + }, + [DROP_TIP_FLOWS.STEPS.CHOOSE_TIP_DROP]: { + allowDoorOpen: false, + }, + [DROP_TIP_FLOWS.STEPS.CHOOSE_BLOWOUT]: { + allowDoorOpen: false, + }, + }, + [ERROR_WHILE_RECOVERING.ROUTE]: { + [ERROR_WHILE_RECOVERING.STEPS.RECOVERY_ACTION_FAILED]: { + allowDoorOpen: false, + }, + [ERROR_WHILE_RECOVERING.STEPS.DROP_TIP_BLOWOUT_FAILED]: { + allowDoorOpen: false, + }, + [ERROR_WHILE_RECOVERING.STEPS.DROP_TIP_TIP_DROP_FAILED]: { + allowDoorOpen: false, + }, + [ERROR_WHILE_RECOVERING.STEPS.DROP_TIP_GENERAL_ERROR]: { + allowDoorOpen: false, + }, + }, + [ROBOT_CANCELING.ROUTE]: { + [ROBOT_CANCELING.STEPS.CANCELING]: { allowDoorOpen: false }, + }, + [ROBOT_IN_MOTION.ROUTE]: { + [ROBOT_IN_MOTION.STEPS.IN_MOTION]: { allowDoorOpen: false }, + }, + [ROBOT_PICKING_UP_TIPS.ROUTE]: { + [ROBOT_PICKING_UP_TIPS.STEPS.PICKING_UP_TIPS]: { + allowDoorOpen: false, + }, + }, + [ROBOT_RESUMING.ROUTE]: { + [ROBOT_RESUMING.STEPS.RESUMING]: { allowDoorOpen: false }, + }, + [ROBOT_RETRYING_STEP.ROUTE]: { + [ROBOT_RETRYING_STEP.STEPS.RETRYING]: { allowDoorOpen: false }, + }, + [ROBOT_SKIPPING_STEP.ROUTE]: { + [ROBOT_SKIPPING_STEP.STEPS.SKIPPING]: { allowDoorOpen: false }, + }, + [ROBOT_DOOR_OPEN.ROUTE]: { + [ROBOT_DOOR_OPEN.STEPS.DOOR_OPEN]: { allowDoorOpen: false }, + }, + [OPTION_SELECTION.ROUTE]: { + [OPTION_SELECTION.STEPS.SELECT]: { allowDoorOpen: false }, + }, + [CANCEL_RUN.ROUTE]: { + [CANCEL_RUN.STEPS.CONFIRM_CANCEL]: { allowDoorOpen: false }, + }, + [IGNORE_AND_SKIP.ROUTE]: { + [IGNORE_AND_SKIP.STEPS.SELECT_IGNORE_KIND]: { + allowDoorOpen: false, + }, + }, + [FILL_MANUALLY_AND_SKIP.ROUTE]: { + [FILL_MANUALLY_AND_SKIP.STEPS.MANUALLY_FILL]: { + allowDoorOpen: true, + }, + [FILL_MANUALLY_AND_SKIP.STEPS.SKIP]: { allowDoorOpen: true }, + }, + [REFILL_AND_RESUME.ROUTE]: {}, + [RETRY_STEP.ROUTE]: { + [RETRY_STEP.STEPS.CONFIRM_RETRY]: { + allowDoorOpen: false, + }, + }, + [RETRY_NEW_TIPS.ROUTE]: { + [RETRY_NEW_TIPS.STEPS.DROP_TIPS]: { allowDoorOpen: false }, + [RETRY_NEW_TIPS.STEPS.REPLACE_TIPS]: { allowDoorOpen: true }, + [RETRY_NEW_TIPS.STEPS.SELECT_TIPS]: { allowDoorOpen: true }, + [RETRY_NEW_TIPS.STEPS.RETRY]: { allowDoorOpen: true }, + }, + [RETRY_SAME_TIPS.ROUTE]: { + [RETRY_SAME_TIPS.STEPS.RETRY]: { allowDoorOpen: true }, + }, + [SKIP_STEP_WITH_NEW_TIPS.ROUTE]: { + [SKIP_STEP_WITH_NEW_TIPS.STEPS.DROP_TIPS]: { + allowDoorOpen: false, + }, + [SKIP_STEP_WITH_NEW_TIPS.STEPS.REPLACE_TIPS]: { + allowDoorOpen: true, + }, + [SKIP_STEP_WITH_NEW_TIPS.STEPS.SELECT_TIPS]: { + allowDoorOpen: true, + }, + [SKIP_STEP_WITH_NEW_TIPS.STEPS.SKIP]: { allowDoorOpen: true }, + }, + [SKIP_STEP_WITH_SAME_TIPS.ROUTE]: { + [SKIP_STEP_WITH_SAME_TIPS.STEPS.SKIP]: { + allowDoorOpen: true, + }, + }, +} as const + export const INVALID = 'INVALID' as const /** diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useCurrentlyRecoveringFrom.test.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useCurrentlyRecoveringFrom.test.ts index 8213d14fec8..4b177972851 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useCurrentlyRecoveringFrom.test.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useCurrentlyRecoveringFrom.test.ts @@ -1,5 +1,6 @@ -import { vi, describe, it, expect } from 'vitest' +import { vi, describe, it, expect, beforeEach } from 'vitest' import { renderHook } from '@testing-library/react' +import { useQueryClient } from 'react-query' import { useCommandQuery } from '@opentrons/react-api-client' import { @@ -10,13 +11,25 @@ import { import { useNotifyAllCommandsQuery } from '/app/resources/runs' import { useCurrentlyRecoveringFrom } from '../useCurrentlyRecoveringFrom' +import type { Mock } from 'vitest' + vi.mock('@opentrons/react-api-client') vi.mock('/app/resources/runs') +vi.mock('react-query') const MOCK_RUN_ID = 'runId' const MOCK_COMMAND_ID = 'commandId' describe('useCurrentlyRecoveringFrom', () => { + let mockInvalidateQueries: Mock + + beforeEach(() => { + mockInvalidateQueries = vi.fn() + vi.mocked(useQueryClient).mockReturnValue({ + invalidateQueries: mockInvalidateQueries, + } as any) + }) + it('disables all queries if the run is not awaiting-recovery', () => { vi.mocked(useNotifyAllCommandsQuery).mockReturnValue({ data: { @@ -97,4 +110,12 @@ describe('useCurrentlyRecoveringFrom', () => { ) expect(result.current).toStrictEqual('mockCommandDetails') }) + + it('calls invalidateQueries when the run enters recovery mode', () => { + renderHook(() => + useCurrentlyRecoveringFrom(MOCK_RUN_ID, RUN_STATUS_AWAITING_RECOVERY) + ) + + expect(mockInvalidateQueries).toHaveBeenCalled() + }) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useErrorName.test.tsx b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useErrorName.test.tsx index 9ed6a072da4..4de37a3a6f5 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useErrorName.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useErrorName.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { describe, it } from 'vitest' import { renderHook, render, screen } from '@testing-library/react' diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryActionMutation.test.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryActionMutation.test.ts index 029f7d5e239..31ba5873ac2 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryActionMutation.test.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryActionMutation.test.ts @@ -1,55 +1,95 @@ import { describe, it, expect, vi, beforeEach } from 'vitest' import { renderHook } from '@testing-library/react' -import { useRunActionMutations } from '@opentrons/react-api-client' +import { usePlayRunMutation } from '@opentrons/react-api-client' import { useRecoveryActionMutation } from '../useRecoveryActionMutation' +import { RECOVERY_MAP } from '../../constants' import type { Mock } from 'vitest' vi.mock('@opentrons/react-api-client', () => ({ - useRunActionMutations: vi.fn(), + usePlayRunMutation: vi.fn(), })) describe('useRecoveryActionMutation', () => { const mockRunId = 'MOCK_ID' - let mockPlayRun: Mock - let mockIsPlayRunActionLoading: boolean + let mockMutateAsync: Mock + let mockIsLoading: boolean + let mockProceedToRouteAndStep: Mock beforeEach(() => { - mockPlayRun = vi.fn() - mockIsPlayRunActionLoading = false + mockMutateAsync = vi.fn() + mockIsLoading = false + mockProceedToRouteAndStep = vi.fn().mockResolvedValue(undefined) - vi.mocked(useRunActionMutations).mockReturnValue({ - playRun: mockPlayRun, - isPlayRunActionLoading: mockIsPlayRunActionLoading, + vi.mocked(usePlayRunMutation).mockReturnValue({ + mutateAsync: mockMutateAsync, + isLoading: mockIsLoading, } as any) }) it('should return resumeRecovery and isResumeRecoveryLoading', () => { - const { result } = renderHook(() => useRecoveryActionMutation(mockRunId)) + const { result } = renderHook(() => + useRecoveryActionMutation(mockRunId, { + proceedToRouteAndStep: mockProceedToRouteAndStep, + } as any) + ) - expect(result.current).toEqual({ - resumeRecovery: mockPlayRun, - isResumeRecoveryLoading: mockIsPlayRunActionLoading, - }) + expect(result.current).toHaveProperty('resumeRecovery') + expect(result.current).toHaveProperty('isResumeRecoveryLoading') + expect(result.current.isResumeRecoveryLoading).toBe(false) }) it('should return updated isResumeRecoveryLoading when it changes', () => { const { result, rerender } = renderHook(() => - useRecoveryActionMutation(mockRunId) + useRecoveryActionMutation(mockRunId, { + proceedToRouteAndStep: mockProceedToRouteAndStep, + } as any) ) expect(result.current.isResumeRecoveryLoading).toBe(false) - mockIsPlayRunActionLoading = true - vi.mocked(useRunActionMutations).mockReturnValue({ - playRun: mockPlayRun, - isPlayRunActionLoading: mockIsPlayRunActionLoading, + mockIsLoading = true + vi.mocked(usePlayRunMutation).mockReturnValue({ + mutateAsync: mockMutateAsync, + isLoading: mockIsLoading, } as any) rerender() expect(result.current.isResumeRecoveryLoading).toBe(true) }) + + it('should call mutateAsync with runId when resumeRecovery is called', async () => { + const { result } = renderHook(() => + useRecoveryActionMutation(mockRunId, { + proceedToRouteAndStep: mockProceedToRouteAndStep, + } as any) + ) + + mockMutateAsync.mockResolvedValue('MOCK_RESULT') + + await result.current.resumeRecovery() + + expect(mockMutateAsync).toHaveBeenCalledWith(mockRunId) + }) + + it('should handle error and proceed to error route when resumeRecovery fails', async () => { + const { result } = renderHook(() => + useRecoveryActionMutation(mockRunId, { + proceedToRouteAndStep: mockProceedToRouteAndStep, + } as any) + ) + + mockMutateAsync.mockRejectedValue(new Error('MOCK_ERROR')) + + await expect(result.current.resumeRecovery()).rejects.toThrow( + 'Could not resume recovery: Error: MOCK_ERROR' + ) + + expect(mockProceedToRouteAndStep).toHaveBeenCalledWith( + RECOVERY_MAP.ERROR_WHILE_RECOVERING.ROUTE + ) + }) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryOptionCopy.test.tsx b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryOptionCopy.test.tsx index 0a0b29e405d..9b013c25b62 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryOptionCopy.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryOptionCopy.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it } from 'vitest' import { screen } from '@testing-library/react' @@ -27,8 +27,8 @@ const render = (props: React.ComponentProps) => { } describe('useRecoveryOptionCopy', () => { - it(`renders the correct copy for ${RECOVERY_MAP.RETRY_FAILED_COMMAND.ROUTE}`, () => { - render({ route: RECOVERY_MAP.RETRY_FAILED_COMMAND.ROUTE }) + it(`renders the correct copy for ${RECOVERY_MAP.RETRY_STEP.ROUTE}`, () => { + render({ route: RECOVERY_MAP.RETRY_STEP.ROUTE }) screen.getByText('Retry step') }) diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryToasts.test.tsx b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryToasts.test.tsx index 8bb0f19b0ce..ea9ef3ae9ba 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryToasts.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryToasts.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, describe, it, expect, beforeEach } from 'vitest' import { I18nextProvider } from 'react-i18next' import { i18n } from '/app/i18n' diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRouteUpdateActions.test.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRouteUpdateActions.test.ts index b8fd72d06df..6e22b929b4c 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRouteUpdateActions.test.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRouteUpdateActions.test.ts @@ -25,10 +25,11 @@ describe('useRouteUpdateActions', () => { hasLaunchedRecovery: true, toggleERWizAsActiveUser: mockToggleERWizard, recoveryMap: { - route: RECOVERY_MAP.RETRY_FAILED_COMMAND.ROUTE, - step: RECOVERY_MAP.RETRY_FAILED_COMMAND.STEPS.CONFIRM_RETRY, + route: RECOVERY_MAP.RETRY_STEP.ROUTE, + step: RECOVERY_MAP.RETRY_STEP.STEPS.CONFIRM_RETRY, }, setRecoveryMap: mockSetRecoveryMap, + doorStatusUtils: { isProhibitedDoorOpen: false, isDoorOpen: false }, } }) @@ -38,7 +39,7 @@ describe('useRouteUpdateActions', () => { ) const { proceedNextStep } = result.current - proceedNextStep() + void proceedNextStep() expect(mockSetRecoveryMap).toHaveBeenCalledWith({ route: OPTION_SELECTION.ROUTE, step: OPTION_SELECTION.STEPS.SELECT, @@ -56,7 +57,7 @@ describe('useRouteUpdateActions', () => { const { proceedNextStep } = result.current - proceedNextStep() + void proceedNextStep() expect(mockToggleERWizard).toHaveBeenCalled() }) @@ -67,7 +68,7 @@ describe('useRouteUpdateActions', () => { ) const { goBackPrevStep } = result.current - goBackPrevStep() + void goBackPrevStep() expect(mockSetRecoveryMap).toHaveBeenCalledWith({ route: OPTION_SELECTION.ROUTE, step: OPTION_SELECTION.STEPS.SELECT, @@ -85,7 +86,7 @@ describe('useRouteUpdateActions', () => { const { goBackPrevStep } = result.current - goBackPrevStep() + void goBackPrevStep() expect(mockToggleERWizard).toHaveBeenCalled() }) @@ -96,56 +97,73 @@ describe('useRouteUpdateActions', () => { ) const { proceedToRouteAndStep } = result.current - proceedToRouteAndStep(RECOVERY_MAP.ROBOT_IN_MOTION.ROUTE) + void proceedToRouteAndStep(RECOVERY_MAP.ROBOT_IN_MOTION.ROUTE) expect(mockSetRecoveryMap).toHaveBeenCalledWith({ route: RECOVERY_MAP.ROBOT_IN_MOTION.ROUTE, step: RECOVERY_MAP.ROBOT_IN_MOTION.STEPS.IN_MOTION, }) }) - it('routes to "robot in motion" when no other motion path is specified', () => { + it('routes to "robot in motion" when no other motion path is specified', async () => { const { result } = renderHook(() => useRouteUpdateActions(useRouteUpdateActionsParams) ) - const { setRobotInMotion } = result.current + const { handleMotionRouting } = result.current - setRobotInMotion(true) + await handleMotionRouting(true) expect(mockSetRecoveryMap).toHaveBeenCalledWith({ route: RECOVERY_MAP.ROBOT_IN_MOTION.ROUTE, step: RECOVERY_MAP.ROBOT_IN_MOTION.STEPS.IN_MOTION, }) }) - it('routes to alternative motion routes if specified', () => { + it('rejects before routing to an "in motion" route if the door is open', async () => { + const params = { + ...useRouteUpdateActionsParams, + doorStatusUtils: { isDoorOpen: true, isProhibitedDoorOpen: false }, + } + + const { result } = renderHook(() => useRouteUpdateActions(params)) + const { handleMotionRouting } = result.current + + await expect(handleMotionRouting(true)).rejects.toThrow() + + expect(mockSetRecoveryMap).toHaveBeenCalledWith({ + route: RECOVERY_MAP.ROBOT_DOOR_OPEN.ROUTE, + step: RECOVERY_MAP.ROBOT_DOOR_OPEN.STEPS.DOOR_OPEN, + }) + }) + + it('routes to alternative motion routes if specified', async () => { const { result } = renderHook(() => useRouteUpdateActions(useRouteUpdateActionsParams) ) - const { setRobotInMotion } = result.current + const { handleMotionRouting } = result.current - setRobotInMotion(true, RECOVERY_MAP.ROBOT_RESUMING.ROUTE) + await handleMotionRouting(true, RECOVERY_MAP.ROBOT_RESUMING.ROUTE) expect(mockSetRecoveryMap).toHaveBeenCalledWith({ route: RECOVERY_MAP.ROBOT_RESUMING.ROUTE, step: RECOVERY_MAP.ROBOT_RESUMING.STEPS.RESUMING, }) }) - it('routes to the route prior to motion after the motion completes', () => { + it('routes to the route prior to motion after the motion completes', async () => { const { result, rerender } = renderHook(() => useRouteUpdateActions(useRouteUpdateActionsParams) ) - const { setRobotInMotion } = result.current + const { handleMotionRouting } = result.current - setRobotInMotion(true) + await handleMotionRouting(true) expect(mockSetRecoveryMap).toHaveBeenCalledWith({ route: RECOVERY_MAP.ROBOT_IN_MOTION.ROUTE, step: RECOVERY_MAP.ROBOT_IN_MOTION.STEPS.IN_MOTION, }) - setRobotInMotion(false) + void handleMotionRouting(false) rerender() expect(mockSetRecoveryMap).toHaveBeenCalledWith({ - route: RECOVERY_MAP.RETRY_FAILED_COMMAND.ROUTE, - step: RECOVERY_MAP.RETRY_FAILED_COMMAND.STEPS.CONFIRM_RETRY, + route: RECOVERY_MAP.RETRY_STEP.ROUTE, + step: RECOVERY_MAP.RETRY_STEP.STEPS.CONFIRM_RETRY, }) }) @@ -155,7 +173,7 @@ describe('useRouteUpdateActions', () => { ) const { proceedToRouteAndStep } = result.current - proceedToRouteAndStep(RECOVERY_MAP.ROBOT_IN_MOTION.ROUTE) + void proceedToRouteAndStep(RECOVERY_MAP.ROBOT_IN_MOTION.ROUTE) expect(mockSetRecoveryMap).toHaveBeenCalledWith({ route: RECOVERY_MAP.ROBOT_IN_MOTION.ROUTE, step: RECOVERY_MAP.ROBOT_IN_MOTION.STEPS.IN_MOTION, @@ -168,13 +186,13 @@ describe('useRouteUpdateActions', () => { ) const { proceedToRouteAndStep } = result.current - proceedToRouteAndStep( - RECOVERY_MAP.RETRY_FAILED_COMMAND.ROUTE, - RECOVERY_MAP.RETRY_FAILED_COMMAND.STEPS.CONFIRM_RETRY + void proceedToRouteAndStep( + RECOVERY_MAP.RETRY_STEP.ROUTE, + RECOVERY_MAP.RETRY_STEP.STEPS.CONFIRM_RETRY ) expect(mockSetRecoveryMap).toHaveBeenCalledWith({ - route: RECOVERY_MAP.RETRY_FAILED_COMMAND.ROUTE, - step: RECOVERY_MAP.RETRY_FAILED_COMMAND.STEPS.CONFIRM_RETRY, + route: RECOVERY_MAP.RETRY_STEP.ROUTE, + step: RECOVERY_MAP.RETRY_STEP.STEPS.CONFIRM_RETRY, }) }) @@ -184,7 +202,10 @@ describe('useRouteUpdateActions', () => { ) const { proceedToRouteAndStep } = result.current - proceedToRouteAndStep(RECOVERY_MAP.ROBOT_IN_MOTION.ROUTE, 'invalid-step') + void proceedToRouteAndStep( + RECOVERY_MAP.ROBOT_IN_MOTION.ROUTE, + 'invalid-step' as any + ) expect(mockSetRecoveryMap).toHaveBeenCalledWith({ route: RECOVERY_MAP.ROBOT_IN_MOTION.ROUTE, step: RECOVERY_MAP.ROBOT_IN_MOTION.STEPS.IN_MOTION, diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useShowDoorInfo.test.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useShowDoorInfo.test.ts index 226ca7de023..a51236144ab 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useShowDoorInfo.test.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useShowDoorInfo.test.ts @@ -7,45 +7,121 @@ import { RUN_STATUS_AWAITING_RECOVERY, } from '@opentrons/api-client' +import { RECOVERY_MAP } from '../../constants' + +import type { IRecoveryMap } from '../../types' + describe('useShowDoorInfo', () => { let initialProps: Parameters[0] + let mockRecoveryMap: IRecoveryMap beforeEach(() => { initialProps = RUN_STATUS_AWAITING_RECOVERY + mockRecoveryMap = { + route: RECOVERY_MAP.OPTION_SELECTION.ROUTE, + step: RECOVERY_MAP.OPTION_SELECTION.STEPS.SELECT, + } as IRecoveryMap }) - it('should return false initially', () => { - const { result } = renderHook(() => useShowDoorInfo(initialProps)) - expect(result.current).toBe(false) + it('should return false values initially', () => { + const { result } = renderHook(() => + useShowDoorInfo(initialProps, mockRecoveryMap) + ) + expect(result.current).toEqual({ + isDoorOpen: false, + isProhibitedDoorOpen: false, + }) }) - it(`should return true when runStatus is ${RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR}`, () => { + it(`should return true values when runStatus is ${RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR}`, () => { const props = RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR - const { result } = renderHook(() => useShowDoorInfo(props)) - expect(result.current).toBe(true) + const { result } = renderHook(() => useShowDoorInfo(props, mockRecoveryMap)) + expect(result.current).toEqual({ + isDoorOpen: true, + isProhibitedDoorOpen: true, + }) }) - it(`should return true when runStatus is ${RUN_STATUS_AWAITING_RECOVERY_PAUSED}`, () => { + it(`should return true values when runStatus is ${RUN_STATUS_AWAITING_RECOVERY_PAUSED}`, () => { const props = RUN_STATUS_AWAITING_RECOVERY_PAUSED - const { result } = renderHook(() => useShowDoorInfo(props)) - expect(result.current).toBe(true) + const { result } = renderHook(() => useShowDoorInfo(props, mockRecoveryMap)) + expect(result.current).toEqual({ + isDoorOpen: true, + isProhibitedDoorOpen: true, + }) }) - it(`should keep returning true when runStatus changes from ${RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR} to ${RUN_STATUS_AWAITING_RECOVERY_PAUSED}`, () => { - const { result, rerender } = renderHook(props => useShowDoorInfo(props), { - initialProps, + it(`should keep returning true values when runStatus changes from ${RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR} to ${RUN_STATUS_AWAITING_RECOVERY_PAUSED}`, () => { + const { result, rerender } = renderHook( + ({ runStatus, recoveryMap }) => useShowDoorInfo(runStatus, recoveryMap), + { + initialProps: { runStatus: initialProps, recoveryMap: mockRecoveryMap }, + } + ) + + act(() => { + rerender({ + runStatus: RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR, + recoveryMap: mockRecoveryMap, + }) + }) + expect(result.current).toEqual({ + isDoorOpen: true, + isProhibitedDoorOpen: true, }) act(() => { - rerender(RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR) + rerender({ + runStatus: RUN_STATUS_AWAITING_RECOVERY_PAUSED, + recoveryMap: mockRecoveryMap, + }) + }) + expect(result.current).toEqual({ + isDoorOpen: true, + isProhibitedDoorOpen: true, + }) + }) + + it('should return false values when runStatus changes to a non-door open status', () => { + const { result, rerender } = renderHook( + ({ runStatus, recoveryMap }) => useShowDoorInfo(runStatus, recoveryMap), + { + initialProps: { + runStatus: RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR, + recoveryMap: mockRecoveryMap, + }, + } + ) + + expect(result.current).toEqual({ + isDoorOpen: true, + isProhibitedDoorOpen: true, }) - expect(result.current).toBe(true) act(() => { - rerender(RUN_STATUS_AWAITING_RECOVERY_PAUSED) + rerender({ + runStatus: RUN_STATUS_AWAITING_RECOVERY as any, + recoveryMap: mockRecoveryMap, + }) }) - expect(result.current).toBe(true) + expect(result.current).toEqual({ + isDoorOpen: false, + isProhibitedDoorOpen: false, + }) + }) + + it('should return false for prohibited door if the route is special-cased even if the door is open', () => { + const props = RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR + + const { result } = renderHook(() => + useShowDoorInfo(props, { + route: RECOVERY_MAP.FILL_MANUALLY_AND_SKIP.ROUTE, + step: RECOVERY_MAP.FILL_MANUALLY_AND_SKIP.STEPS.MANUALLY_FILL, + }) + ) + + expect(result.current.isProhibitedDoorOpen).toEqual(false) }) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/index.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/index.ts index 4a97b82afc1..2411c95c30e 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/index.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/index.ts @@ -1,13 +1,13 @@ export { useCurrentlyRecoveringFrom } from './useCurrentlyRecoveringFrom' export { useErrorMessage } from './useErrorMessage' export { useErrorName } from './useErrorName' -export { useShowDoorInfo } from './useShowDoorInfo' export { useRecoveryCommands } from './useRecoveryCommands' export { useRouteUpdateActions } from './useRouteUpdateActions' export { useERUtils } from './useERUtils' export { useRecoveryTakeover } from './useRecoveryTakeover' export { useRetainedFailedCommandBySource } from './useRetainedFailedCommandBySource' +export type { ERUtilsProps } from './useERUtils' export type { UseRouteUpdateActionsResult } from './useRouteUpdateActions' export type { UseRecoveryCommandsResult } from './useRecoveryCommands' export type { RecoveryTipStatusUtils } from './useRecoveryTipStatus' @@ -15,3 +15,4 @@ export type { ERUtilsResults } from './useERUtils' export type { UseFailedLabwareUtilsResult } from './useFailedLabwareUtils' export type { UseRecoveryTakeoverResult } from './useRecoveryTakeover' export type { FailedCommandBySource } from './useRetainedFailedCommandBySource' +export type { UseRecoveryRoutingResult } from './useRecoveryRouting' diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useCurrentlyRecoveringFrom.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useCurrentlyRecoveringFrom.ts index 319ba718270..d0273643b61 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useCurrentlyRecoveringFrom.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useCurrentlyRecoveringFrom.ts @@ -1,9 +1,12 @@ +import { useEffect } from 'react' +import { useQueryClient } from 'react-query' + import { RUN_STATUS_AWAITING_RECOVERY, RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR, RUN_STATUS_AWAITING_RECOVERY_PAUSED, } from '@opentrons/api-client' -import { useCommandQuery } from '@opentrons/react-api-client' +import { useCommandQuery, useHost } from '@opentrons/react-api-client' import { useNotifyAllCommandsQuery } from '/app/resources/runs' @@ -25,10 +28,19 @@ export function useCurrentlyRecoveringFrom( runId: string, runStatus: RunStatus | null ): FailedCommand | null { + const queryClient = useQueryClient() + const host = useHost() // There can only be a currentlyRecoveringFrom command when the run is in recovery mode. // In case we're falling back to polling, only enable queries when that is the case. const isRunInRecoveryMode = VALID_RECOVERY_FETCH_STATUSES.includes(runStatus) + // Prevent stale data on subsequent recoveries by clearing the query cache at the start of each recovery. + useEffect(() => { + if (isRunInRecoveryMode) { + void queryClient.invalidateQueries([host, 'runs', runId]) + } + }, [isRunInRecoveryMode, host, runId]) + const { data: allCommandsQueryData } = useNotifyAllCommandsQuery( runId, { cursor: null, pageLength: 0 }, // pageLength 0 because we only care about the links. diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useDeckMapUtils.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useDeckMapUtils.ts index 3c8a45faabb..93caceaf4c2 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useDeckMapUtils.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useDeckMapUtils.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useMemo } from 'react' import { getDeckDefFromRobotType, @@ -50,7 +50,7 @@ export function useDeckMapUtils({ const deckConfig = getSimplestDeckConfigForProtocol(protocolAnalysis) const deckDef = getDeckDefFromRobotType(robotType) - const currentModulesInfo = React.useMemo( + const currentModulesInfo = useMemo( () => getRunCurrentModulesInfo({ runRecord, @@ -60,7 +60,7 @@ export function useDeckMapUtils({ [runRecord, deckDef, protocolAnalysis] ) - const runCurrentModules = React.useMemo( + const runCurrentModules = useMemo( () => getRunCurrentModulesOnDeck({ failedLabwareUtils, @@ -69,12 +69,12 @@ export function useDeckMapUtils({ [runId, protocolAnalysis, runRecord, deckDef, failedLabwareUtils] ) - const currentLabwareInfo = React.useMemo( + const currentLabwareInfo = useMemo( () => getRunCurrentLabwareInfo({ runRecord, protocolAnalysis }), [runRecord, protocolAnalysis] ) - const runCurrentLabware = React.useMemo( + const runCurrentLabware = useMemo( () => getRunCurrentLabwareOnDeck({ failedLabwareUtils, diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useERUtils.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useERUtils.ts index fdb7ca0ad22..b571bcd890d 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useERUtils.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useERUtils.ts @@ -5,7 +5,11 @@ import { useRecoveryCommands } from './useRecoveryCommands' import { useRecoveryTipStatus } from './useRecoveryTipStatus' import { useRecoveryRouting } from './useRecoveryRouting' import { useFailedLabwareUtils } from './useFailedLabwareUtils' -import { getFailedCommandPipetteInfo, getNextSteps } from '../utils' +import { + getFailedCommandPipetteInfo, + getNextSteps, + cleanupRecoveryState, +} from '../utils' import { useDeckMapUtils } from './useDeckMapUtils' import { useNotifyAllCommandsQuery, @@ -16,6 +20,7 @@ import { useRecoveryActionMutation } from './useRecoveryActionMutation' import { useRunningStepCounts } from '/app/resources/protocols/hooks' import { useRecoveryToasts } from './useRecoveryToasts' import { useRecoveryAnalytics } from '/app/redux-resources/analytics' +import { useShowDoorInfo } from './useShowDoorInfo' import type { PipetteData } from '@opentrons/api-client' import type { RobotType } from '@opentrons/shared-data' @@ -35,6 +40,7 @@ import type { StepCounts } from '/app/resources/protocols/hooks' import type { UseRecoveryAnalyticsResult } from '/app/redux-resources/analytics' import type { UseRecoveryTakeoverResult } from './useRecoveryTakeover' import type { useRetainedFailedCommandBySource } from './useRetainedFailedCommandBySource' +import type { UseShowDoorInfoResult } from './useShowDoorInfo' export type ERUtilsProps = Omit & { toggleERWizAsActiveUser: UseRecoveryTakeoverResult['toggleERWizAsActiveUser'] @@ -42,12 +48,13 @@ export type ERUtilsProps = Omit & { isOnDevice: boolean robotType: RobotType failedCommand: ReturnType + showTakeover: boolean } export interface ERUtilsResults { recoveryMap: IRecoveryMap currentRecoveryOptionUtils: CurrentRecoveryOptionUtils - routeUpdateActions: UseRouteUpdateActionsResult + routeUpdateActions: Omit recoveryCommands: UseRecoveryCommandsResult tipStatusUtils: RecoveryTipStatusUtils failedLabwareUtils: UseFailedLabwareUtilsResult @@ -60,6 +67,7 @@ export interface ERUtilsResults { commandsAfterFailedCommand: ReturnType subMapUtils: SubMapUtils analytics: UseRecoveryAnalyticsResult + doorStatusUtils: UseShowDoorInfoResult } const SUBSEQUENT_COMMAND_DEPTH = 2 @@ -72,6 +80,8 @@ export function useERUtils({ protocolAnalysis, isOnDevice, robotType, + runStatus, + showTakeover, }: ERUtilsProps): ERUtilsResults { const { data: attachedInstruments } = useInstrumentsQuery() const { data: runRecord } = useNotifyRunQuery(runId) @@ -97,6 +107,8 @@ export function useERUtils({ ...subMapUtils } = useRecoveryRouting() + const doorStatusUtils = useShowDoorInfo(runStatus, recoveryMap) + const recoveryToastUtils = useRecoveryToasts({ currentStepCount: stepCounts.currentStepNumber, selectedRecoveryOption: currentRecoveryOptionUtils.selectedRecoveryOption, @@ -123,6 +135,7 @@ export function useERUtils({ recoveryMap, toggleERWizAsActiveUser, setRecoveryMap: setRM, + doorStatusUtils, }) const failedLabwareUtils = useFailedLabwareUtils({ @@ -150,7 +163,10 @@ export function useERUtils({ failedLabwareUtils, }) - const recoveryActionMutationUtils = useRecoveryActionMutation(runId) + const recoveryActionMutationUtils = useRecoveryActionMutation( + runId, + routeUpdateActions + ) // TODO(jh, 06-14-24): Ensure other string build utilities that are internal to ErrorRecoveryFlows are exported under // one utility object in useERUtils. @@ -160,6 +176,13 @@ export function useERUtils({ protocolAnalysis, SUBSEQUENT_COMMAND_DEPTH ) + + cleanupRecoveryState({ + isTakeover: showTakeover, + setRM, + stashedMapRef: routeUpdateActions.stashedMapRef, + }) + return { recoveryMap, subMapUtils, @@ -176,5 +199,6 @@ export function useERUtils({ stepCounts, commandsAfterFailedCommand, analytics, + doorStatusUtils, } } diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useErrorName.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useErrorName.ts index bde35d89fbf..9827ce202a8 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useErrorName.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useErrorName.ts @@ -17,8 +17,9 @@ export function useErrorName(errorKind: ErrorKind): string { return t('pipette_overpressure') case ERROR_KINDS.OVERPRESSURE_WHILE_DISPENSING: return t('pipette_overpressure') - // The only "general error" case currently is tipPhysicallyMissing. - default: + case ERROR_KINDS.TIP_NOT_DETECTED: return t('tip_not_detected') + default: + return t('error') } } diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useFailedLabwareUtils.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useFailedLabwareUtils.ts index b5f2866a154..092e066b432 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useFailedLabwareUtils.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useFailedLabwareUtils.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useMemo, useState } from 'react' import without from 'lodash/without' import { @@ -56,7 +56,7 @@ export function useFailedLabwareUtils({ runCommands, runRecord, }: UseFailedLabwareUtilsProps): UseFailedLabwareUtilsResult { - const recentRelevantFailedLabwareCmd = React.useMemo( + const recentRelevantFailedLabwareCmd = useMemo( () => getRelevantFailedLabwareCmdFrom({ failedCommandByRunRecord, @@ -67,7 +67,7 @@ export function useFailedLabwareUtils({ const tipSelectionUtils = useTipSelectionUtils(recentRelevantFailedLabwareCmd) - const failedLabwareDetails = React.useMemo( + const failedLabwareDetails = useMemo( () => getFailedCmdRelevantLabware( protocolAnalysis, @@ -77,7 +77,7 @@ export function useFailedLabwareUtils({ [protocolAnalysis?.id, recentRelevantFailedLabwareCmd?.key] ) - const failedLabware = React.useMemo( + const failedLabware = useMemo( () => getFailedLabware(recentRelevantFailedLabwareCmd, runRecord), [recentRelevantFailedLabwareCmd?.key] ) @@ -185,7 +185,7 @@ interface UseTipSelectionUtilsResult { function useTipSelectionUtils( recentRelevantFailedLabwareCmd: FailedCommandRelevantLabware ): UseTipSelectionUtilsResult { - const [selectedLocs, setSelectedLocs] = React.useState(null) + const [selectedLocs, setSelectedLocs] = useState(null) const initialLocs = useInitialSelectedLocationsFrom( recentRelevantFailedLabwareCmd @@ -211,7 +211,7 @@ function useTipSelectionUtils( } // Use this labware to represent all tip racks for manual tip selection. - const tipSelectorDef = React.useMemo( + const tipSelectorDef = useMemo( () => getAllLabwareDefs().thermoscientificnunc96Wellplate1300UlV1, [] ) @@ -232,7 +232,7 @@ function useTipSelectionUtils( function useInitialSelectedLocationsFrom( recentRelevantFailedLabwareCmd: FailedCommandRelevantLabware ): WellGroup | null { - const [initialWells, setInitialWells] = React.useState(null) + const [initialWells, setInitialWells] = useState(null) // Note that while other commands may have a wellName associated with them, // we are only interested in wells for the purposes of tip picking up. diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryActionMutation.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryActionMutation.ts index 5f33df7941d..0d4bdd83b42 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryActionMutation.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryActionMutation.ts @@ -1,20 +1,35 @@ +import { usePlayRunMutation } from '@opentrons/react-api-client' + +import { RECOVERY_MAP } from '../constants' + +import type { RunAction } from '@opentrons/api-client' +import type { ERUtilsResults } from './useERUtils' import type { ErrorRecoveryFlowsProps } from '..' -import { useRunActionMutations } from '@opentrons/react-api-client' export interface RecoveryActionMutationResult { - resumeRecovery: ReturnType['playRun'] - isResumeRecoveryLoading: ReturnType< - typeof useRunActionMutations - >['isPlayRunActionLoading'] + resumeRecovery: () => Promise + isResumeRecoveryLoading: ReturnType['isLoading'] } export function useRecoveryActionMutation( - runId: ErrorRecoveryFlowsProps['runId'] + runId: ErrorRecoveryFlowsProps['runId'], + routeUpdateActions: ERUtilsResults['routeUpdateActions'] ): RecoveryActionMutationResult { const { - playRun: resumeRecovery, - isPlayRunActionLoading: isResumeRecoveryLoading, - } = useRunActionMutations(runId) + mutateAsync, + isLoading: isResumeRecoveryLoading, + } = usePlayRunMutation() + const { proceedToRouteAndStep } = routeUpdateActions + + const resumeRecovery = (): Promise => { + return mutateAsync(runId).catch(e => { + return proceedToRouteAndStep( + RECOVERY_MAP.ERROR_WHILE_RECOVERING.ROUTE + ).then(() => { + throw new Error(`Could not resume recovery: ${e}`) + }) + }) + } return { resumeRecovery, isResumeRecoveryLoading } } diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts index ddf5eda3bbf..b57b5cc9aa2 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useCallback } from 'react' import head from 'lodash/head' import { @@ -82,6 +82,21 @@ export function useRecoveryCommands({ const { updateErrorRecoveryPolicy } = useUpdateErrorRecoveryPolicy(runId) const { makeSuccessToast } = recoveryToastUtils + const chainRunRecoveryCommands = useCallback( + ( + commands: CreateCommand[], + continuePastFailure: boolean = false + ): Promise => + chainRunCommands(commands, continuePastFailure).catch(e => { + console.warn(`Error executing "fixit" command: ${e}`) + analytics.reportActionSelectedResult(selectedRecoveryOption, 'failed') + // the catch never occurs if continuePastCommandFailure is "true" + void proceedToRouteAndStep(RECOVERY_MAP.ERROR_WHILE_RECOVERING.ROUTE) + return Promise.reject(new Error(`Could not execute command: ${e}`)) + }), + [analytics, selectedRecoveryOption] + ) + const buildRetryPrepMove = (): MoveToCoordinatesCreateCommand | null => { type InPlaceCommand = | AspirateInPlaceRunTimeCommand @@ -126,22 +141,8 @@ export function useRecoveryCommands({ : null : null } - const chainRunRecoveryCommands = React.useCallback( - ( - commands: CreateCommand[], - continuePastFailure: boolean = false - ): Promise => - chainRunCommands(commands, continuePastFailure).catch(e => { - console.warn(`Error executing "fixit" command: ${e}`) - analytics.reportActionSelectedResult(selectedRecoveryOption, 'failed') - // the catch never occurs if continuePastCommandFailure is "true" - void proceedToRouteAndStep(RECOVERY_MAP.ERROR_WHILE_RECOVERING.ROUTE) - return Promise.reject(new Error(`Could not execute command: ${e}`)) - }), - [chainRunCommands] - ) - const retryFailedCommand = React.useCallback((): Promise => { + const retryFailedCommand = useCallback((): Promise => { const { commandType, params } = failedCommandByRunRecord as FailedCommand // Null case is handled before command could be issued. return chainRunRecoveryCommands( [ @@ -153,12 +154,12 @@ export function useRecoveryCommands({ }, [chainRunRecoveryCommands, failedCommandByRunRecord?.key]) // Homes the Z-axis of all attached pipettes. - const homePipetteZAxes = React.useCallback((): Promise => { + const homePipetteZAxes = useCallback((): Promise => { return chainRunRecoveryCommands([HOME_PIPETTE_Z_AXES]) }, [chainRunRecoveryCommands]) // Pick up the user-selected tips - const pickUpTips = React.useCallback((): Promise => { + const pickUpTips = useCallback((): Promise => { const { selectedTipLocations, failedLabware } = failedLabwareUtils const pickUpTipCmd = buildPickUpTips( @@ -174,26 +175,26 @@ export function useRecoveryCommands({ } }, [chainRunRecoveryCommands, failedCommandByRunRecord, failedLabwareUtils]) - const resumeRun = React.useCallback((): void => { + const resumeRun = useCallback((): void => { void resumeRunFromRecovery(runId).then(() => { analytics.reportActionSelectedResult(selectedRecoveryOption, 'succeeded') makeSuccessToast() }) }, [runId, resumeRunFromRecovery, makeSuccessToast]) - const cancelRun = React.useCallback((): void => { + const cancelRun = useCallback((): void => { analytics.reportActionSelectedResult(selectedRecoveryOption, 'succeeded') stopRun(runId) }, [runId]) - const skipFailedCommand = React.useCallback((): void => { + const skipFailedCommand = useCallback((): void => { void resumeRunFromRecovery(runId).then(() => { analytics.reportActionSelectedResult(selectedRecoveryOption, 'succeeded') makeSuccessToast() }) }, [runId, resumeRunFromRecovery, makeSuccessToast]) - const ignoreErrorKindThisRun = React.useCallback((): Promise => { + const ignoreErrorKindThisRun = useCallback((): Promise => { if (failedCommandByRunRecord?.error != null) { const ignorePolicyRules = buildIgnorePolicyRules( failedCommandByRunRecord.commandType, diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryOptionCopy.tsx b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryOptionCopy.tsx index 4e636f6fd78..47b7782c094 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryOptionCopy.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryOptionCopy.tsx @@ -14,7 +14,7 @@ export function useRecoveryOptionCopy(): ( recoveryOption: RecoveryRoute | null ): string => { switch (recoveryOption) { - case RECOVERY_MAP.RETRY_FAILED_COMMAND.ROUTE: + case RECOVERY_MAP.RETRY_STEP.ROUTE: return t('retry_step') case RECOVERY_MAP.CANCEL_RUN.ROUTE: return t('cancel_run') diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryRouting.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryRouting.ts index b97a1206739..cbf23a1393b 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryRouting.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryRouting.ts @@ -1,15 +1,15 @@ -import * as React from 'react' +import { useState } from 'react' import { RECOVERY_MAP } from '../constants' -import type { IRecoveryMap, RecoveryRoute, ValidSubMap } from '../types' +import type { IRecoveryMap, RecoveryRoute, ValidDropTipSubMap } from '../types' // Utils for getting/setting the current submap. See useRecoveryRouting. export interface SubMapUtils { /* See useRecoveryRouting. */ - updateSubMap: (subMap: ValidSubMap | null) => void + updateSubMap: (subMap: ValidDropTipSubMap | null) => void /* See useRecoveryRouting. */ - subMap: ValidSubMap | null + subMap: ValidDropTipSubMap | null } export interface UseRecoveryRoutingResult { @@ -28,12 +28,12 @@ export interface UseRecoveryRoutingResult { * */ export function useRecoveryRouting(): UseRecoveryRoutingResult { - const [recoveryMap, setRecoveryMap] = React.useState({ + const [recoveryMap, setRecoveryMap] = useState({ route: RECOVERY_MAP.OPTION_SELECTION.ROUTE, step: RECOVERY_MAP.OPTION_SELECTION.STEPS.SELECT, }) - const [subMap, setSubMap] = React.useState(null) + const [subMap, setSubMap] = useState(null) const currentRecoveryOptionUtils = useSelectedRecoveryOption() @@ -56,7 +56,7 @@ export function useSelectedRecoveryOption(): CurrentRecoveryOptionUtils { const [ selectedRecoveryOption, setSelectedRecoveryOption, - ] = React.useState(null) + ] = useState(null) return { selectedRecoveryOption, diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryTakeover.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryTakeover.ts index 417b6da1719..8e8b3466039 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryTakeover.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryTakeover.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { useSelector } from 'react-redux' import { getUserId } from '/app/redux/config' @@ -40,7 +40,7 @@ export interface UseRecoveryTakeoverResult { export function useRecoveryTakeover( toggleERWiz: UseERWizardResult['toggleERWizard'] ): UseRecoveryTakeoverResult { - const [isActiveUser, setIsActiveUser] = React.useState(false) + const [isActiveUser, setIsActiveUser] = useState(false) const thisUserId = useSelector(getUserId) const { userId: activeId, intent } = useClientDataRecovery({ @@ -49,14 +49,14 @@ export function useRecoveryTakeover( const { updateWithIntent, clearClientData } = useUpdateClientDataRecovery() // Update the client's active user status implicitly if revoked by a different client. - React.useEffect(() => { + useEffect(() => { if (isActiveUser && activeId !== thisUserId) { setIsActiveUser(false) } }, [activeId]) // Not all dependencies added for intended behavior! // If Error Recovery unrenders and this client is the active user, revoke the client's active user status. - React.useEffect(() => { + useEffect(() => { return () => { if (isActiveUser) { clearClientData() diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryTipStatus.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryTipStatus.ts index 505ba6aff7c..698b8f6cf22 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryTipStatus.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryTipStatus.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import head from 'lodash/head' import { useHost } from '@opentrons/react-api-client' @@ -28,11 +28,11 @@ export type RecoveryTipStatusUtils = TipAttachmentStatusResult & { export function useRecoveryTipStatus( props: UseRecoveryTipStatusProps ): RecoveryTipStatusUtils { - const [isLoadingTipStatus, setIsLoadingTipStatus] = React.useState(false) + const [isLoadingTipStatus, setIsLoadingTipStatus] = useState(false) const [ failedCommandPipette, setFailedCommandPipette, - ] = React.useState(null) + ] = useState(null) const host = useHost() const tipAttachmentStatusUtils = useTipAttachmentStatus({ diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryToasts.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryToasts.ts index ea63447c78f..0ab5d0806c0 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryToasts.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryToasts.ts @@ -154,7 +154,7 @@ function handleRecoveryOptionAction( case RECOVERY_MAP.CANCEL_RUN.ROUTE: case RECOVERY_MAP.RETRY_SAME_TIPS.ROUTE: case RECOVERY_MAP.RETRY_NEW_TIPS.ROUTE: - case RECOVERY_MAP.RETRY_FAILED_COMMAND.ROUTE: + case RECOVERY_MAP.RETRY_STEP.ROUTE: return currentStepReturnVal default: return 'HANDLE RECOVERY TOAST OPTION EXPLICITLY.' diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRetainedFailedCommandBySource.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useRetainedFailedCommandBySource.ts index 10231f5e0cc..c967d4968b1 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useRetainedFailedCommandBySource.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRetainedFailedCommandBySource.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import type { RunTimeCommand } from '@opentrons/shared-data' import type { ErrorRecoveryFlowsProps } from '..' @@ -25,9 +25,9 @@ export function useRetainedFailedCommandBySource( const [ retainedFailedCommand, setRetainedFailedCommand, - ] = React.useState(null) + ] = useState(null) - React.useEffect(() => { + useEffect(() => { if (failedCommandByRunRecord !== null) { const failedCommandByAnalysis = protocolAnalysis?.commands.find( diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRouteUpdateActions.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useRouteUpdateActions.ts index 74c8186bf79..8a5c19ce318 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useRouteUpdateActions.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRouteUpdateActions.ts @@ -1,9 +1,10 @@ -import * as React from 'react' +import type { MutableRefObject } from 'react' +import { useRef, useCallback } from 'react' import last from 'lodash/last' + import head from 'lodash/head' import { INVALID, RECOVERY_MAP, STEP_ORDER } from '../constants' - import type { IRecoveryMap, RecoveryRoute, @@ -11,12 +12,14 @@ import type { RouteStep, } from '../types' import type { UseRecoveryTakeoverResult } from './useRecoveryTakeover' +import type { UseShowDoorInfoResult } from './useShowDoorInfo' export interface GetRouteUpdateActionsParams { hasLaunchedRecovery: boolean toggleERWizAsActiveUser: UseRecoveryTakeoverResult['toggleERWizAsActiveUser'] recoveryMap: IRecoveryMap setRecoveryMap: (recoveryMap: IRecoveryMap) => void + doorStatusUtils: UseShowDoorInfoResult } export interface UseRouteUpdateActionsResult { @@ -29,23 +32,33 @@ export interface UseRouteUpdateActionsResult { route: RecoveryRoute, step?: RouteStep ) => Promise - /* Stashes the current map then sets the current map to robot in motion. Restores the map after motion completes. */ - setRobotInMotion: ( + /* Stashes the current map then sets the current map to robot in motion after validating the door is closed. + Restores the map after motion completes. */ + handleMotionRouting: ( inMotion: boolean, movingRoute?: RobotMovingRoute ) => Promise + /* Contains the recovery map prior to implicit redirection, if any. Example, if the user is on route A, step A, and the + app implicitly navigates the user to route Z, step Z, the stashed map will contain route A, step A. */ + stashedMap: IRecoveryMap | null + stashedMapRef: MutableRefObject } // Utilities related to routing within the error recovery flows. export function useRouteUpdateActions( routeUpdateActionsParams: GetRouteUpdateActionsParams ): UseRouteUpdateActionsResult { - const { recoveryMap, setRecoveryMap } = routeUpdateActionsParams + const { + recoveryMap, + setRecoveryMap, + doorStatusUtils, + } = routeUpdateActionsParams const { route: currentRoute, step: currentStep } = recoveryMap - const { OPTION_SELECTION, ROBOT_IN_MOTION } = RECOVERY_MAP - const stashedMapRef = React.useRef(null) + const { OPTION_SELECTION, ROBOT_IN_MOTION, ROBOT_DOOR_OPEN } = RECOVERY_MAP + const { isDoorOpen } = doorStatusUtils + const stashedMapRef = useRef(null) - const goBackPrevStep = React.useCallback((): Promise => { + const goBackPrevStep = useCallback((): Promise => { return new Promise((resolve, reject) => { const { getPrevStep } = getRecoveryRouteNavigation(currentRoute) const updatedStep = getPrevStep(currentStep) @@ -60,7 +73,7 @@ export function useRouteUpdateActions( }) }, [currentStep, currentRoute, routeUpdateActionsParams]) - const proceedNextStep = React.useCallback((): Promise => { + const proceedNextStep = useCallback((): Promise => { return new Promise((resolve, reject) => { const { getNextStep } = getRecoveryRouteNavigation(currentRoute) const updatedStep = getNextStep(currentStep) @@ -75,7 +88,7 @@ export function useRouteUpdateActions( }) }, [currentStep, currentRoute, routeUpdateActionsParams]) - const proceedToRouteAndStep = React.useCallback( + const proceedToRouteAndStep = useCallback( (route: RecoveryRoute, step?: RouteStep): Promise => { return new Promise((resolve, reject) => { const newFlowSteps = STEP_ORDER[route] @@ -90,7 +103,30 @@ export function useRouteUpdateActions( [] ) - const setRobotInMotion = React.useCallback( + // If the door is permitted on the current step, but the robot is about to move, we need to manually redirect users + // to the door modal. + const checkDoorStatus = useCallback((): Promise => { + return new Promise((resolve, reject) => { + if (isDoorOpen) { + stashedMapRef.current = { route: currentRoute, step: currentStep } + + setRecoveryMap({ + route: ROBOT_DOOR_OPEN.ROUTE, + step: ROBOT_DOOR_OPEN.STEPS.DOOR_OPEN, + }) + + reject( + new Error( + 'Cannot perform a command while the door is open. Routing to door open modal.' + ) + ) + } else { + resolve() + } + }) + }, [currentRoute, currentStep, isDoorOpen]) + + const setRobotInMotion = useCallback( (inMotion: boolean, robotMovingRoute?: RobotMovingRoute): Promise => { return new Promise((resolve, reject) => { if (inMotion) { @@ -122,11 +158,27 @@ export function useRouteUpdateActions( [currentRoute, currentStep] ) + const handleMotionRouting = ( + inMotion: boolean, + robotMovingRoute?: RobotMovingRoute + ): Promise => { + // Only check door status if we are fixin' to move. + if (inMotion) { + return checkDoorStatus().then(() => + setRobotInMotion(inMotion, robotMovingRoute) + ) + } else { + return setRobotInMotion(inMotion, robotMovingRoute) + } + } + return { goBackPrevStep, proceedNextStep, proceedToRouteAndStep, - setRobotInMotion, + handleMotionRouting, + stashedMap: stashedMapRef.current, + stashedMapRef: stashedMapRef, } } @@ -167,7 +219,7 @@ export function getRecoveryRouteNavigation( } type DetermineRecoveryRoutingParams = GetRouteUpdateActionsParams & { - updatedStep: string + updatedStep: RouteStep currentRoute: RecoveryRoute } diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useShowDoorInfo.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useShowDoorInfo.ts index f73f1a22ae0..61b9131b15e 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useShowDoorInfo.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useShowDoorInfo.ts @@ -1,38 +1,55 @@ -import * as React from 'react' - import { RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR, RUN_STATUS_AWAITING_RECOVERY_PAUSED, } from '@opentrons/api-client' +import { RECOVERY_MAP_METADATA } from '../constants' + import type { RunStatus } from '@opentrons/api-client' import type { ErrorRecoveryFlowsProps } from '../index' +import type { IRecoveryMap } from '../types' const DOOR_OPEN_STATUSES: RunStatus[] = [ RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR, RUN_STATUS_AWAITING_RECOVERY_PAUSED, ] -// Whether the door is open or the user has not yet resumed the run after a door open event. +export interface UseShowDoorInfoResult { + /* Whether the door actually open, regardless of whether a door open event is prohibited . */ + isDoorOpen: boolean + /* Whether the door is open and prohibited to be open. */ + isProhibitedDoorOpen: boolean +} + +// Whether the door is open and not permitted to be open or the user has not yet resumed the run after a door open event. export function useShowDoorInfo( - runStatus: ErrorRecoveryFlowsProps['runStatus'] -): boolean { - const [showDoorModal, setShowDoorModal] = React.useState(false) - - React.useEffect(() => { - // TODO(jh, 07-16-24): "recovery paused" is only used for door status and therefore - // a valid way to ensure all apps show the door open prompt, however this could be problematic in the future. - // Consider restructuring this check once the takeover modals are added. - if (runStatus != null && DOOR_OPEN_STATUSES.includes(runStatus)) { - setShowDoorModal(true) - } else if ( - showDoorModal && - runStatus != null && - !DOOR_OPEN_STATUSES.includes(runStatus) - ) { - setShowDoorModal(false) + runStatus: ErrorRecoveryFlowsProps['runStatus'], + recoveryMap: IRecoveryMap +): UseShowDoorInfoResult { + // TODO(jh, 07-16-24): "recovery paused" is only used for door status and therefore + // a valid way to ensure all apps show the door open prompt, however this could be problematic in the future. + // Consider restructuring this check once the takeover modals are added. + const isDoorOpen = runStatus != null && DOOR_OPEN_STATUSES.includes(runStatus) + const isProhibitedDoorOpen = isDoorOpen && !isDoorPermittedOpen(recoveryMap) + + return { isDoorOpen, isProhibitedDoorOpen } +} + +function isDoorPermittedOpen(recoveryMap: IRecoveryMap): boolean { + const { route, step } = recoveryMap + + if (route in RECOVERY_MAP_METADATA) { + const routeConfig = RECOVERY_MAP_METADATA[route] + + if (step in routeConfig) { + // @ts-expect-error Overly nested type that TS struggles to infer. + return routeConfig[step].allowDoorOpen } - }, [runStatus, showDoorModal]) + } - return showDoorModal + console.error( + 'Unexpected route/step combination when attempting to get door metadata for recovery map: ', + recoveryMap + ) + return false } diff --git a/app/src/organisms/ErrorRecoveryFlows/index.tsx b/app/src/organisms/ErrorRecoveryFlows/index.tsx index 85ae5c806fa..fce380ced9f 100644 --- a/app/src/organisms/ErrorRecoveryFlows/index.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useSelector } from 'react-redux' import { @@ -18,14 +18,13 @@ import { useHost } from '@opentrons/react-api-client' import { getIsOnDevice } from '/app/redux/config' import { ErrorRecoveryWizard, useERWizard } from './ErrorRecoveryWizard' -import { RunPausedSplash, useRunPausedSplash } from './RunPausedSplash' +import { RecoverySplash, useRecoverySplash } from './RecoverySplash' import { RecoveryTakeover } from './RecoveryTakeover' import { useCurrentlyRecoveringFrom, useERUtils, useRecoveryTakeover, useRetainedFailedCommandBySource, - useShowDoorInfo, } from './hooks' import type { RunStatus } from '@opentrons/api-client' @@ -58,11 +57,9 @@ export function useErrorRecoveryFlows( runId: string, runStatus: RunStatus | null ): UseErrorRecoveryResult { - const [isERActive, setIsERActive] = React.useState(false) + const [isERActive, setIsERActive] = useState(false) // If client accesses a valid ER runs status besides AWAITING_RECOVERY but accesses it outside of Error Recovery flows, don't show ER. - const [hasSeenAwaitingRecovery, setHasSeenAwaitingRecovery] = React.useState( - false - ) + const [hasSeenAwaitingRecovery, setHasSeenAwaitingRecovery] = useState(false) const failedCommand = useCurrentlyRecoveringFrom(runId, runStatus) if ( @@ -128,15 +125,12 @@ export function ErrorRecoveryFlows( const robotType = protocolAnalysis?.robotType ?? OT2_ROBOT_TYPE const robotName = useHost()?.robotName ?? 'robot' - const isDoorOpen = useShowDoorInfo(runStatus) const { showTakeover, isActiveUser, intent, toggleERWizAsActiveUser, } = useRecoveryTakeover(toggleERWizard) - const renderWizard = isActiveUser && (showERWizard || isDoorOpen) - const showSplash = useRunPausedSplash(isOnDevice, renderWizard) const recoveryUtils = useERUtils({ ...props, @@ -144,9 +138,15 @@ export function ErrorRecoveryFlows( toggleERWizAsActiveUser, isOnDevice, robotType, + showTakeover, failedCommand: failedCommandBySource, }) + const renderWizard = + isActiveUser && + (showERWizard || recoveryUtils.doorStatusUtils.isProhibitedDoorOpen) + const showSplash = useRecoverySplash(isOnDevice, renderWizard as boolean) + return ( <> {showTakeover ? ( @@ -163,12 +163,11 @@ export function ErrorRecoveryFlows( {...recoveryUtils} robotType={robotType} isOnDevice={isOnDevice} - isDoorOpen={isDoorOpen} failedCommand={failedCommandBySource} /> ) : null} {showSplash ? ( - ) : null} diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/ErrorDetailsModal.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/ErrorDetailsModal.tsx index ba898d94d7e..e81c4c2106b 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/ErrorDetailsModal.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/ErrorDetailsModal.tsx @@ -27,7 +27,7 @@ import type { IconProps } from '@opentrons/components' import type { OddModalHeaderBaseProps } from '/app/molecules/OddModal/types' import type { ERUtilsResults, useRetainedFailedCommandBySource } from '../hooks' import type { ErrorRecoveryFlowsProps } from '..' -import type { DesktopSizeType } from '../types' +import type { DesktopSizeType, ErrorKind } from '../types' export function useErrorDetailsModal(): { showModal: boolean @@ -59,11 +59,12 @@ export function ErrorDetailsModal(props: ErrorDetailsModalProps): JSX.Element { const errorKind = getErrorKind(failedCommand?.byRunRecord ?? null) const errorName = useErrorName(errorKind) - const getIsOverpressureErrorKind = (): boolean => { + const isNotificationErrorKind = (): boolean => { switch (errorKind) { case ERROR_KINDS.OVERPRESSURE_PREPARE_TO_ASPIRATE: case ERROR_KINDS.OVERPRESSURE_WHILE_ASPIRATING: case ERROR_KINDS.OVERPRESSURE_WHILE_DISPENSING: + case ERROR_KINDS.TIP_NOT_DETECTED: return true default: return false @@ -83,7 +84,9 @@ export function ErrorDetailsModal(props: ErrorDetailsModalProps): JSX.Element { toggleModal={toggleModal} modalHeader={modalHeader} > - {getIsOverpressureErrorKind() ? : null} + {isNotificationErrorKind() ? ( + + ) : null} , getTopPortalEl() ) @@ -94,7 +97,9 @@ export function ErrorDetailsModal(props: ErrorDetailsModalProps): JSX.Element { toggleModal={toggleModal} modalHeader={modalHeader} > - {getIsOverpressureErrorKind() ? : null} + {isNotificationErrorKind() ? ( + + ) : null} , getModalPortalEl() ) @@ -191,14 +196,48 @@ export function ErrorDetailsModalODD( ) } -export function OverpressureBanner(): JSX.Element | null { +export function NotificationBanner({ + errorKind, +}: { + errorKind: ErrorKind +}): JSX.Element { + const buildContent = (): JSX.Element => { + switch (errorKind) { + case ERROR_KINDS.OVERPRESSURE_PREPARE_TO_ASPIRATE: + case ERROR_KINDS.OVERPRESSURE_WHILE_ASPIRATING: + case ERROR_KINDS.OVERPRESSURE_WHILE_DISPENSING: + return + case ERROR_KINDS.TIP_NOT_DETECTED: + return + default: + console.error('Handle error kind notification banners explicitly.') + return
+ } + } + + return buildContent() +} + +export function OverpressureBanner(): JSX.Element { const { t } = useTranslation('error_recovery') return ( + ) +} + +export function TipNotDetectedBanner(): JSX.Element { + const { t } = useTranslation('error_recovery') + + return ( + ) } diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/FailedStepNextStep.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/FailedStepNextStep.tsx index 8261e313848..bad7b536dfe 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/FailedStepNextStep.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/FailedStepNextStep.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { CategorizedStepContent } from '/app/molecules/InterventionModal' import type { RecoveryContentProps } from '../types' diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/LeftColumnLabwareInfo.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/LeftColumnLabwareInfo.tsx index 756165ed425..80c0422a940 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/LeftColumnLabwareInfo.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/LeftColumnLabwareInfo.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { InterventionContent } from '/app/molecules/InterventionModal/InterventionContent' diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryContentWrapper.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryContentWrapper.tsx index 53a66630047..9274079897f 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryContentWrapper.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryContentWrapper.tsx @@ -1,7 +1,7 @@ // TODO: replace this by making these props true of interventionmodal content wrappers // once error recovery uses interventionmodal consistently -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { DIRECTION_COLUMN, diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryFooterButtons.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryFooterButtons.tsx index b9aa7503baa..37e5cf54dde 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryFooterButtons.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryFooterButtons.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryInterventionModal.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryInterventionModal.tsx index 901b5708974..2d73bba9e22 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryInterventionModal.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryInterventionModal.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { createPortal } from 'react-dom' import { css } from 'styled-components' diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryRadioGroup.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryRadioGroup.tsx index 571f0b0333a..d200123d0b3 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryRadioGroup.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryRadioGroup.tsx @@ -1,6 +1,4 @@ -import * as React from 'react' - -import type { ChangeEventHandler } from 'react' +import type { ChangeEventHandler, ReactNode, ComponentProps } from 'react' import { RadioGroup, SPACING, Flex } from '@opentrons/components' // note: this typescript stuff is so that e.currentTarget.value in the ChangeEventHandler @@ -12,12 +10,12 @@ export interface Target extends Omit { export type Options = Array<{ value: T - children: React.ReactNode + children: ReactNode }> export interface RecoveryRadioGroupProps extends Omit< - React.ComponentProps, + ComponentProps, 'labelTextClassName' | 'options' | 'onchange' > { options: Options diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/ReplaceTips.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/ReplaceTips.tsx index 08e600ae4be..936980dad80 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/ReplaceTips.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/ReplaceTips.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - import { Flex } from '@opentrons/components' import { useTranslation } from 'react-i18next' diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/SelectTips.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/SelectTips.tsx index 21688dbf41c..0f55a5abe79 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/SelectTips.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/SelectTips.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { RECOVERY_MAP } from '../constants' @@ -23,14 +23,14 @@ export function SelectTips(props: RecoveryContentProps): JSX.Element | null { const { pickUpTips } = recoveryCommands const { goBackPrevStep, - setRobotInMotion, + handleMotionRouting, proceedNextStep, } = routeUpdateActions const { t } = useTranslation('error_recovery') - const [showTipSelectModal, setShowTipSelectModal] = React.useState(false) + const [showTipSelectModal, setShowTipSelectModal] = useState(false) const primaryBtnOnClick = (): Promise => { - return setRobotInMotion(true, ROBOT_PICKING_UP_TIPS.ROUTE) + return handleMotionRouting(true, ROBOT_PICKING_UP_TIPS.ROUTE) .then(() => pickUpTips()) .then(() => proceedNextStep()) } diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/StepInfo.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/StepInfo.tsx index 002ba1458f8..cd9d0c3d00e 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/StepInfo.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/StepInfo.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/TipSelection.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/TipSelection.tsx index 2b9084cb0f4..cde82286c2f 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/TipSelection.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/TipSelection.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - import { WellSelection } from '../../WellSelection' import type { WellGroup } from '@opentrons/components' diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/TipSelectionModal.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/TipSelectionModal.tsx index 6fe9106b1c4..57a5ac4c678 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/TipSelectionModal.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/TipSelectionModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { createPortal } from 'react-dom' diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/TwoColTextAndFailedStepNextStep.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/TwoColTextAndFailedStepNextStep.tsx index e5b69c4cc67..64702965d73 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/TwoColTextAndFailedStepNextStep.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/TwoColTextAndFailedStepNextStep.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { DIRECTION_COLUMN, diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/ErrorDetailsModal.test.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/ErrorDetailsModal.test.tsx index 6a88cdc8ccc..e06baba1a80 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/ErrorDetailsModal.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/ErrorDetailsModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, beforeEach, expect, vi } from 'vitest' import { screen, act, renderHook } from '@testing-library/react' @@ -12,6 +12,7 @@ import { useErrorDetailsModal, ErrorDetailsModal, OverpressureBanner, + TipNotDetectedBanner, } from '../ErrorDetailsModal' vi.mock('react-dom', () => ({ @@ -92,7 +93,7 @@ describe('ErrorDetailsModal', () => { }) IS_ODD.forEach(isOnDevice => { - it('renders the OverpressureBanner when the error kind is an overpressure error', () => { + it('renders an inline banner when the error kind is an overpressure error', () => { props.failedCommand = { ...props.failedCommand, byRunRecord: { @@ -106,9 +107,23 @@ describe('ErrorDetailsModal', () => { screen.getByText('MOCK_INLINE_NOTIFICATION') }) - it('does not render the OverpressureBanner when the error kind is not an overpressure error', () => { + it('renders an inline banner when the error kind is a tip not detected error', () => { + props.failedCommand = { + ...props.failedCommand, + byRunRecord: { + ...props.failedCommand?.byRunRecord, + commandType: 'pickUpTip', + error: { isDefined: true, errorType: 'tipPhysicallyMissing' }, + }, + } as any render({ ...props, isOnDevice }) + screen.getByText('MOCK_INLINE_NOTIFICATION') + }) + + it('does not render a banner when the error kind is not explicitly handled', () => { + render({ ...props, isOnDevice, failedCommand: {} as any }) + expect(screen.queryByText('MOCK_INLINE_NOTIFICATION')).toBeNull() }) }) @@ -137,3 +152,27 @@ describe('OverpressureBanner', () => { ) }) }) + +describe('TipNotDetectedBanner', () => { + beforeEach(() => { + vi.mocked(InlineNotification).mockReturnValue( +
MOCK_INLINE_NOTIFICATION
+ ) + }) + + it('renders the InlineNotification', () => { + renderWithProviders(, { + i18nInstance: i18n, + }) + expect(vi.mocked(InlineNotification)).toHaveBeenCalledWith( + expect.objectContaining({ + type: 'alert', + heading: + 'Tip presence errors are usually caused by improperly placed labware or inaccurate labware offsets', + message: + ' If the issue persists, cancel the run and initiate Labware Position Check', + }), + {} + ) + }) +}) diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/LeftColumnLabwareInfo.test.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/LeftColumnLabwareInfo.test.tsx index 6a070e62e77..c714c0bc8a2 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/LeftColumnLabwareInfo.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/LeftColumnLabwareInfo.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, beforeEach, expect, vi } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/RecoveryFooterButtons.test.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/RecoveryFooterButtons.test.tsx index 3520851ac0b..6381d2b579a 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/RecoveryFooterButtons.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/RecoveryFooterButtons.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, describe, it, expect, beforeEach } from 'vitest' import { screen, fireEvent } from '@testing-library/react' diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/SelectTips.test.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/SelectTips.test.tsx index 87526dfe8c0..88fd55e982d 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/SelectTips.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/SelectTips.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, expect, beforeEach } from 'vitest' import { screen, fireEvent, waitFor } from '@testing-library/react' @@ -24,13 +24,13 @@ const render = (props: React.ComponentProps) => { describe('SelectTips', () => { let props: React.ComponentProps let mockGoBackPrevStep: Mock - let mockSetRobotInMotion: Mock + let mockhandleMotionRouting: Mock let mockProceedNextStep: Mock let mockPickUpTips: Mock beforeEach(() => { mockGoBackPrevStep = vi.fn() - mockSetRobotInMotion = vi.fn(() => Promise.resolve()) + mockhandleMotionRouting = vi.fn(() => Promise.resolve()) mockProceedNextStep = vi.fn() mockPickUpTips = vi.fn(() => Promise.resolve()) @@ -38,7 +38,7 @@ describe('SelectTips', () => { ...mockRecoveryContentProps, routeUpdateActions: { goBackPrevStep: mockGoBackPrevStep, - setRobotInMotion: mockSetRobotInMotion, + handleMotionRouting: mockhandleMotionRouting, proceedNextStep: mockProceedNextStep, } as any, recoveryCommands: { @@ -69,7 +69,7 @@ describe('SelectTips', () => { }) it('calls the correct routeUpdateActions and recoveryCommands in the correct order when the primary button is clicked', async () => { - const setRobotInMotionMock = vi.fn(() => Promise.resolve()) + const handleMotionRoutingMock = vi.fn(() => Promise.resolve()) const pickUpTipsMock = vi.fn(() => Promise.resolve()) const proceedNextStepMock = vi.fn() @@ -78,7 +78,7 @@ describe('SelectTips', () => { } as any const mockRouteUpdateActions = { - setRobotInMotion: setRobotInMotionMock, + handleMotionRouting: handleMotionRoutingMock, proceedNextStep: proceedNextStepMock, } as any @@ -92,10 +92,10 @@ describe('SelectTips', () => { fireEvent.click(primaryBtn) await waitFor(() => { - expect(setRobotInMotionMock).toHaveBeenCalledTimes(1) + expect(handleMotionRoutingMock).toHaveBeenCalledTimes(1) }) await waitFor(() => { - expect(setRobotInMotionMock).toHaveBeenCalledWith( + expect(handleMotionRoutingMock).toHaveBeenCalledWith( true, RECOVERY_MAP.ROBOT_PICKING_UP_TIPS.ROUTE ) @@ -107,7 +107,7 @@ describe('SelectTips', () => { expect(proceedNextStepMock).toHaveBeenCalledTimes(1) }) - expect(setRobotInMotionMock.mock.invocationCallOrder[0]).toBeLessThan( + expect(handleMotionRoutingMock.mock.invocationCallOrder[0]).toBeLessThan( pickUpTipsMock.mock.invocationCallOrder[0] ) diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/StepInfo.test.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/StepInfo.test.tsx index 9a305804c6c..d6fbb50c345 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/StepInfo.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/StepInfo.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, beforeEach, vi } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/TipSelection.test.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/TipSelection.test.tsx index 96da4e8144d..d8f48776a9c 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/TipSelection.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/TipSelection.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, expect, beforeEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/TipSelectionModal.test.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/TipSelectionModal.test.tsx index 50ed13e34d4..fed5a44d4ce 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/TipSelectionModal.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/TipSelectionModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, beforeEach, expect } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/ErrorRecoveryFlows/types.ts b/app/src/organisms/ErrorRecoveryFlows/types.ts index f3df4a86c50..53a5fc64e62 100644 --- a/app/src/organisms/ErrorRecoveryFlows/types.ts +++ b/app/src/organisms/ErrorRecoveryFlows/types.ts @@ -1,14 +1,46 @@ import type { RunCommandSummary } from '@opentrons/api-client' -import type { ERROR_KINDS, RECOVERY_MAP, INVALID } from './constants' +import type { ERROR_KINDS, INVALID, RECOVERY_MAP } from './constants' import type { ErrorRecoveryWizardProps } from './ErrorRecoveryWizard' import type { DropTipFlowsRoute, DropTipFlowsStep, } from '../DropTipWizardFlows/types' +/** + * Misc Recovery Types + */ export type FailedCommand = RunCommandSummary +export type ErrorKind = typeof ERROR_KINDS[keyof typeof ERROR_KINDS] + +/** + * Prop Specific Types + */ +export type RecoveryContentProps = ErrorRecoveryWizardProps & { + errorKind: ErrorKind + isOnDevice: boolean +} + +/** + * Drop Tip Specific Types + */ +export type ValidDropTipSubRoutes = DropTipFlowsRoute +export type ValidDropTipSubSteps = DropTipFlowsStep +export interface ValidDropTipSubMap { + route: ValidDropTipSubRoutes + step: ValidDropTipSubSteps | null +} + +/** + * Recovery Map Types + */ +export interface IRecoveryMap { + route: RecoveryRoute + step: RouteStep +} +type RecoveryMap = typeof RECOVERY_MAP export type InvalidStep = typeof INVALID -export type RecoveryRoute = typeof RECOVERY_MAP[keyof typeof RECOVERY_MAP]['ROUTE'] +export type RouteKey = RecoveryMap[keyof RecoveryMap]['ROUTE'] +export type RecoveryRoute = RouteKey export type RobotMovingRoute = | typeof RECOVERY_MAP['ROBOT_IN_MOTION']['ROUTE'] | typeof RECOVERY_MAP['ROBOT_RESUMING']['ROUTE'] @@ -16,63 +48,35 @@ export type RobotMovingRoute = | typeof RECOVERY_MAP['ROBOT_CANCELING']['ROUTE'] | typeof RECOVERY_MAP['ROBOT_PICKING_UP_TIPS']['ROUTE'] | typeof RECOVERY_MAP['ROBOT_SKIPPING_STEP']['ROUTE'] -export type ErrorKind = typeof ERROR_KINDS[keyof typeof ERROR_KINDS] -interface RecoveryMapDetails { - ROUTE: string - STEPS: Record - STEP_ORDER: RouteStep -} - -export type ValidSubRoutes = DropTipFlowsRoute -export type ValidSubSteps = DropTipFlowsStep -export interface ValidSubMap { - route: ValidSubRoutes - step: ValidSubSteps | null -} +type OriginalRouteKey = keyof RecoveryMap +type StepsForRoute = RecoveryMap[{ + [K in OriginalRouteKey]: RecoveryMap[K]['ROUTE'] extends R ? K : never +}[OriginalRouteKey]]['STEPS'] +type StepKey = StepsForRoute[keyof StepsForRoute] +export type AllStepTypes = { + [R in RouteKey]: StepKey +}[RouteKey] -export type RecoveryMap = Record +export type RouteStep = AllStepTypes | InvalidStep export type StepOrder = { [K in RecoveryRoute]: RouteStep[] } -type RecoveryStep< - K extends keyof RecoveryMap -> = RecoveryMap[K]['STEPS'][keyof RecoveryMap[K]['STEPS']] - -type RobotCancellingRunStep = RecoveryStep<'ROBOT_CANCELING'> -type RobotInMotionStep = RecoveryStep<'ROBOT_IN_MOTION'> -type RobotResumingStep = RecoveryStep<'ROBOT_RESUMING'> -type RobotRetryingCommandStep = RecoveryStep<'ROBOT_RETRYING_COMMAND'> -type BeforeBeginningStep = RecoveryStep<'BEFORE_BEGINNING'> -type CancelRunStep = RecoveryStep<'CANCEL_RUN'> -type DropTipStep = RecoveryStep<'DROP_TIP'> -type IgnoreAndResumeStep = RecoveryStep<'IGNORE_AND_RESUME'> -type RefillAndResumeStep = RecoveryStep<'REFILL_AND_RESUME'> -type ResumeStep = RecoveryStep<'RESUME'> -type OptionSelectionStep = RecoveryStep<'OPTION_SELECTION'> - -export type RouteStep = - | RobotInMotionStep - | RobotResumingStep - | RobotRetryingCommandStep - | BeforeBeginningStep - | CancelRunStep - | DropTipStep - | IgnoreAndResumeStep - | ResumeStep - | OptionSelectionStep - | RefillAndResumeStep - | RobotCancellingRunStep - -export interface IRecoveryMap { - route: RecoveryRoute - step: RouteStep +/** + * Route/Step Properties Types + */ +interface StepDoorConfig { + allowDoorOpen: boolean } - -export type RecoveryContentProps = ErrorRecoveryWizardProps & { - errorKind: ErrorKind - isOnDevice: boolean +type RouteDoorConfig = { + [Step in StepKey & string]: StepDoorConfig +} +export type RecoveryRouteStepMetadata = { + [R in RouteKey]: RouteDoorConfig } +/** + * Style Types + */ export type DesktopSizeType = 'desktop-small' | 'desktop-large' diff --git a/app/src/organisms/ErrorRecoveryFlows/utils/__tests__/cleanupRecoveryState.test.ts b/app/src/organisms/ErrorRecoveryFlows/utils/__tests__/cleanupRecoveryState.test.ts new file mode 100644 index 00000000000..0bb17054c11 --- /dev/null +++ b/app/src/organisms/ErrorRecoveryFlows/utils/__tests__/cleanupRecoveryState.test.ts @@ -0,0 +1,56 @@ +import { describe, it, expect, beforeEach, vi } from 'vitest' + +import { cleanupRecoveryState } from '../cleanupRecoveryState' +import { RECOVERY_MAP } from '../../constants' + +describe('cleanupRecoveryState', () => { + let props: Parameters[0] + let mockSetRM: ReturnType + + beforeEach(() => { + mockSetRM = vi.fn() + props = { + isTakeover: false, + stashedMapRef: { + current: { + route: RECOVERY_MAP.FILL_MANUALLY_AND_SKIP.ROUTE, + step: RECOVERY_MAP.FILL_MANUALLY_AND_SKIP.STEPS.SKIP, + }, + }, + setRM: mockSetRM, + } + }) + + it('does not modify state when isTakeover is false', () => { + cleanupRecoveryState(props) + + expect(props.stashedMapRef.current).toEqual({ + route: RECOVERY_MAP.FILL_MANUALLY_AND_SKIP.ROUTE, + step: RECOVERY_MAP.FILL_MANUALLY_AND_SKIP.STEPS.SKIP, + }) + expect(mockSetRM).not.toHaveBeenCalled() + }) + + it('resets state when isTakeover is true', () => { + props.isTakeover = true + cleanupRecoveryState(props) + + expect(props.stashedMapRef.current).toBeNull() + expect(mockSetRM).toHaveBeenCalledWith({ + route: RECOVERY_MAP.OPTION_SELECTION.ROUTE, + step: RECOVERY_MAP.OPTION_SELECTION.STEPS.SELECT, + }) + }) + + it('handles case when stashedMapRef.current is already null', () => { + props.isTakeover = true + props.stashedMapRef.current = null + cleanupRecoveryState(props) + + expect(props.stashedMapRef.current).toBeNull() + expect(mockSetRM).toHaveBeenCalledWith({ + route: RECOVERY_MAP.OPTION_SELECTION.ROUTE, + step: RECOVERY_MAP.OPTION_SELECTION.STEPS.SELECT, + }) + }) +}) diff --git a/app/src/organisms/ErrorRecoveryFlows/utils/__tests__/getErrorKind.test.ts b/app/src/organisms/ErrorRecoveryFlows/utils/__tests__/getErrorKind.test.ts index adad317fd2a..162c29d2566 100644 --- a/app/src/organisms/ErrorRecoveryFlows/utils/__tests__/getErrorKind.test.ts +++ b/app/src/organisms/ErrorRecoveryFlows/utils/__tests__/getErrorKind.test.ts @@ -39,6 +39,17 @@ describe('getErrorKind', () => { expect(result).toEqual(ERROR_KINDS.OVERPRESSURE_WHILE_DISPENSING) }) + it(`returns ${ERROR_KINDS.TIP_NOT_DETECTED} for ${DEFINED_ERROR_TYPES.TIP_PHYSICALLY_MISSING} errorType`, () => { + const result = getErrorKind({ + commandType: 'pickUpTip', + error: { + isDefined: true, + errorType: DEFINED_ERROR_TYPES.TIP_PHYSICALLY_MISSING, + } as RunCommandError, + } as RunTimeCommand) + expect(result).toEqual(ERROR_KINDS.TIP_NOT_DETECTED) + }) + it(`returns ${ERROR_KINDS.GENERAL_ERROR} for undefined errors`, () => { const result = getErrorKind({ commandType: 'aspirate', diff --git a/app/src/organisms/ErrorRecoveryFlows/utils/cleanupRecoveryState.ts b/app/src/organisms/ErrorRecoveryFlows/utils/cleanupRecoveryState.ts new file mode 100644 index 00000000000..fddc29c1983 --- /dev/null +++ b/app/src/organisms/ErrorRecoveryFlows/utils/cleanupRecoveryState.ts @@ -0,0 +1,29 @@ +import { RECOVERY_MAP } from '../constants' + +import type { + ERUtilsProps, + UseRouteUpdateActionsResult, + UseRecoveryRoutingResult, +} from '../hooks' + +export interface UseCleanupProps { + isTakeover: ERUtilsProps['showTakeover'] + stashedMapRef: UseRouteUpdateActionsResult['stashedMapRef'] + setRM: UseRecoveryRoutingResult['setRM'] +} + +// When certain events (ex, a takeover) occur, reset state that needs to be reset. +export function cleanupRecoveryState({ + isTakeover, + stashedMapRef, + setRM, +}: UseCleanupProps): void { + if (isTakeover) { + stashedMapRef.current = null + + setRM({ + route: RECOVERY_MAP.OPTION_SELECTION.ROUTE, + step: RECOVERY_MAP.OPTION_SELECTION.STEPS.SELECT, + }) + } +} diff --git a/app/src/organisms/ErrorRecoveryFlows/utils/getErrorKind.ts b/app/src/organisms/ErrorRecoveryFlows/utils/getErrorKind.ts index bc16e11619c..89475f34c93 100644 --- a/app/src/organisms/ErrorRecoveryFlows/utils/getErrorKind.ts +++ b/app/src/organisms/ErrorRecoveryFlows/utils/getErrorKind.ts @@ -16,18 +16,24 @@ export function getErrorKind(failedCommand: RunTimeCommand | null): ErrorKind { if ( commandType === 'aspirate' && errorType === DEFINED_ERROR_TYPES.OVERPRESSURE - ) + ) { return ERROR_KINDS.OVERPRESSURE_WHILE_ASPIRATING - else if ( + } else if ( commandType === 'dispense' && errorType === DEFINED_ERROR_TYPES.OVERPRESSURE - ) + ) { return ERROR_KINDS.OVERPRESSURE_WHILE_DISPENSING - else if ( + } else if ( commandType === 'liquidProbe' && errorType === DEFINED_ERROR_TYPES.LIQUID_NOT_FOUND - ) + ) { return ERROR_KINDS.NO_LIQUID_DETECTED + } else if ( + commandType === 'pickUpTip' && + errorType === DEFINED_ERROR_TYPES.TIP_PHYSICALLY_MISSING + ) { + return ERROR_KINDS.TIP_NOT_DETECTED + } // todo(mm, 2024-07-02): Also handle aspirateInPlace and dispenseInPlace. // https://opentrons.atlassian.net/browse/EXEC-593 } diff --git a/app/src/organisms/ErrorRecoveryFlows/utils/index.ts b/app/src/organisms/ErrorRecoveryFlows/utils/index.ts index 53195d5d5fa..42f0a40d6f1 100644 --- a/app/src/organisms/ErrorRecoveryFlows/utils/index.ts +++ b/app/src/organisms/ErrorRecoveryFlows/utils/index.ts @@ -1,3 +1,4 @@ export { getErrorKind } from './getErrorKind' export { getFailedCommandPipetteInfo } from './getFailedCommandPipetteInfo' export { getNextStep, getNextSteps } from './getNextStep' +export { cleanupRecoveryState } from './cleanupRecoveryState' diff --git a/app/src/organisms/FirmwareUpdateModal/FirmwareUpdateTakeover.tsx b/app/src/organisms/FirmwareUpdateModal/FirmwareUpdateTakeover.tsx index d2cf969fbda..615165a6f26 100644 --- a/app/src/organisms/FirmwareUpdateModal/FirmwareUpdateTakeover.tsx +++ b/app/src/organisms/FirmwareUpdateModal/FirmwareUpdateTakeover.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { createPortal } from 'react-dom' import { @@ -16,19 +16,18 @@ import type { Subsystem, InstrumentData } from '@opentrons/api-client' const POLL_INTERVAL_MS = 5000 export function FirmwareUpdateTakeover(): JSX.Element { - const [ - showUpdateNeededModal, - setShowUpdateNeededModal, - ] = React.useState(false) + const [showUpdateNeededModal, setShowUpdateNeededModal] = useState( + false + ) const [ initiatedSubsystemUpdate, setInitiatedSubsystemUpdate, - ] = React.useState(null) + ] = useState(null) const instrumentsData = useInstrumentsQuery({ refetchInterval: POLL_INTERVAL_MS, }).data?.data - const [instrumentsToUpdate, setInstrumentsToUpdate] = React.useState< + const [instrumentsToUpdate, setInstrumentsToUpdate] = useState< InstrumentData[] >([]) instrumentsData?.forEach(instrument => { @@ -41,7 +40,7 @@ export function FirmwareUpdateTakeover(): JSX.Element { setInstrumentsToUpdate([...instrumentsToUpdate, instrument]) } }) - const [indexToUpdate, setIndexToUpdate] = React.useState(0) + const [indexToUpdate, setIndexToUpdate] = useState(0) const { data: maintenanceRunData } = useNotifyCurrentMaintenanceRun({ refetchInterval: POLL_INTERVAL_MS, @@ -63,7 +62,7 @@ export function FirmwareUpdateTakeover(): JSX.Element { externalSubsystemUpdate?.id ?? null ) - React.useEffect(() => { + useEffect(() => { // in case instruments are updated elsewhere in the app, clear update needed list // when all instruments are ok but array has elements if ( diff --git a/app/src/organisms/FirmwareUpdateModal/UpdateInProgressModal.tsx b/app/src/organisms/FirmwareUpdateModal/UpdateInProgressModal.tsx index 9fa84bf49ba..a870f97bc67 100644 --- a/app/src/organisms/FirmwareUpdateModal/UpdateInProgressModal.tsx +++ b/app/src/organisms/FirmwareUpdateModal/UpdateInProgressModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' import { diff --git a/app/src/organisms/FirmwareUpdateModal/UpdateNeededModal.tsx b/app/src/organisms/FirmwareUpdateModal/UpdateNeededModal.tsx index de00b12655d..fa925c11e0b 100644 --- a/app/src/organisms/FirmwareUpdateModal/UpdateNeededModal.tsx +++ b/app/src/organisms/FirmwareUpdateModal/UpdateNeededModal.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { createPortal } from 'react-dom' import { useTranslation, Trans } from 'react-i18next' import capitalize from 'lodash/capitalize' @@ -34,9 +34,9 @@ interface UpdateNeededModalProps { export function UpdateNeededModal(props: UpdateNeededModalProps): JSX.Element { const { onClose, shouldExit, subsystem, setInitiatedSubsystemUpdate } = props const { t } = useTranslation('firmware_update') - const [updateId, setUpdateId] = React.useState(null) + const [updateId, setUpdateId] = useState(null) // when we move to the next subsystem to update, set updateId back to null - React.useEffect(() => { + useEffect(() => { setUpdateId(null) }, [subsystem]) @@ -58,7 +58,7 @@ export function UpdateNeededModal(props: UpdateNeededModalProps): JSX.Element { const status = updateData?.data.updateStatus const ongoingUpdateId = updateData?.data.id - React.useEffect(() => { + useEffect(() => { if (status === 'done') { setInitiatedSubsystemUpdate(null) } diff --git a/app/src/organisms/FirmwareUpdateModal/UpdateResultsModal.tsx b/app/src/organisms/FirmwareUpdateModal/UpdateResultsModal.tsx index 19c540e82f0..e305d451c7f 100644 --- a/app/src/organisms/FirmwareUpdateModal/UpdateResultsModal.tsx +++ b/app/src/organisms/FirmwareUpdateModal/UpdateResultsModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation, Trans } from 'react-i18next' import { ALIGN_CENTER, @@ -13,7 +12,7 @@ import { } from '@opentrons/components' import { SmallButton } from '/app/atoms/buttons' import { OddModal } from '/app/molecules/OddModal' -import { usePipetteModelSpecs } from '/app/resources/instruments/hooks' +import { usePipetteModelSpecs } from '/app/local-resources/instruments' import type { InstrumentData, PipetteData } from '@opentrons/api-client' import type { OddModalHeaderBaseProps } from '/app/molecules/OddModal/types' diff --git a/app/src/organisms/FirmwareUpdateModal/__tests__/FirmwareUpdateModal.test.tsx b/app/src/organisms/FirmwareUpdateModal/__tests__/FirmwareUpdateModal.test.tsx index 08a0c8f6aad..bb205f3f852 100644 --- a/app/src/organisms/FirmwareUpdateModal/__tests__/FirmwareUpdateModal.test.tsx +++ b/app/src/organisms/FirmwareUpdateModal/__tests__/FirmwareUpdateModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { act, screen, waitFor } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/FirmwareUpdateModal/__tests__/FirmwareUpdateTakeover.test.tsx b/app/src/organisms/FirmwareUpdateModal/__tests__/FirmwareUpdateTakeover.test.tsx index ca2ee5a9d90..6d619555834 100644 --- a/app/src/organisms/FirmwareUpdateModal/__tests__/FirmwareUpdateTakeover.test.tsx +++ b/app/src/organisms/FirmwareUpdateModal/__tests__/FirmwareUpdateTakeover.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/FirmwareUpdateModal/__tests__/UpdateInProgressModal.test.tsx b/app/src/organisms/FirmwareUpdateModal/__tests__/UpdateInProgressModal.test.tsx index 496da7421fb..f2ce6047481 100644 --- a/app/src/organisms/FirmwareUpdateModal/__tests__/UpdateInProgressModal.test.tsx +++ b/app/src/organisms/FirmwareUpdateModal/__tests__/UpdateInProgressModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/FirmwareUpdateModal/__tests__/UpdateNeededModal.test.tsx b/app/src/organisms/FirmwareUpdateModal/__tests__/UpdateNeededModal.test.tsx index 545854dbf1e..53c91223b47 100644 --- a/app/src/organisms/FirmwareUpdateModal/__tests__/UpdateNeededModal.test.tsx +++ b/app/src/organisms/FirmwareUpdateModal/__tests__/UpdateNeededModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/FirmwareUpdateModal/__tests__/UpdateResultsModal.test.tsx b/app/src/organisms/FirmwareUpdateModal/__tests__/UpdateResultsModal.test.tsx index 47095ebdc5e..29c5233db45 100644 --- a/app/src/organisms/FirmwareUpdateModal/__tests__/UpdateResultsModal.test.tsx +++ b/app/src/organisms/FirmwareUpdateModal/__tests__/UpdateResultsModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/FirmwareUpdateModal/index.tsx b/app/src/organisms/FirmwareUpdateModal/index.tsx index cb449588ea1..0ed40409797 100644 --- a/app/src/organisms/FirmwareUpdateModal/index.tsx +++ b/app/src/organisms/FirmwareUpdateModal/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { css } from 'styled-components' import { ALIGN_CENTER, @@ -73,8 +73,8 @@ export const FirmwareUpdateModal = ( description, isOnDevice, } = props - const [updateId, setUpdateId] = React.useState(null) - const [firmwareText, setFirmwareText] = React.useState(null) + const [updateId, setUpdateId] = useState(null) + const [firmwareText, setFirmwareText] = useState(null) const { data: attachedInstruments, refetch: refetchInstruments, @@ -92,7 +92,7 @@ export const FirmwareUpdateModal = ( (i): i is BadGripper | BadPipette => !i.ok && i.subsystem === subsystem ) ?? false - React.useEffect(() => { + useEffect(() => { setTimeout(() => { if (!updateNeeded) { setFirmwareText(proceedDescription) @@ -107,7 +107,7 @@ export const FirmwareUpdateModal = ( const { data: updateData } = useSubsystemUpdateQuery(updateId) const status = updateData?.data.updateStatus - React.useEffect(() => { + useEffect(() => { if ((status != null || updateNeeded) && firmwareText !== description) { setFirmwareText(description) } diff --git a/app/src/organisms/GripperCard/AboutGripperSlideout.tsx b/app/src/organisms/GripperCard/AboutGripperSlideout.tsx index 5cfb59e2c21..a95a409f759 100644 --- a/app/src/organisms/GripperCard/AboutGripperSlideout.tsx +++ b/app/src/organisms/GripperCard/AboutGripperSlideout.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { COLORS, diff --git a/app/src/organisms/GripperCard/__tests__/AboutGripperSlideout.test.tsx b/app/src/organisms/GripperCard/__tests__/AboutGripperSlideout.test.tsx index fa6fff2a552..9ad4381a40c 100644 --- a/app/src/organisms/GripperCard/__tests__/AboutGripperSlideout.test.tsx +++ b/app/src/organisms/GripperCard/__tests__/AboutGripperSlideout.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen, fireEvent } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/GripperCard/__tests__/GripperCard.test.tsx b/app/src/organisms/GripperCard/__tests__/GripperCard.test.tsx index 32f523b82ca..40051f51bca 100644 --- a/app/src/organisms/GripperCard/__tests__/GripperCard.test.tsx +++ b/app/src/organisms/GripperCard/__tests__/GripperCard.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/GripperCard/index.tsx b/app/src/organisms/GripperCard/index.tsx index fd720a99e0f..34ade2ce899 100644 --- a/app/src/organisms/GripperCard/index.tsx +++ b/app/src/organisms/GripperCard/index.tsx @@ -2,6 +2,7 @@ import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { css } from 'styled-components' import { + Banner, CURSOR_POINTER, LegacyStyledText, SPACING, @@ -9,7 +10,6 @@ import { } from '@opentrons/components' import { getGripperDisplayName } from '@opentrons/shared-data' import { useCurrentSubsystemUpdateQuery } from '@opentrons/react-api-client' -import { Banner } from '/app/atoms/Banner' import { InstrumentCard } from '/app/molecules/InstrumentCard' import { GripperWizardFlows } from '../GripperWizardFlows' import { AboutGripperSlideout } from './AboutGripperSlideout' diff --git a/app/src/organisms/GripperWizardFlows/BeforeBeginning.tsx b/app/src/organisms/GripperWizardFlows/BeforeBeginning.tsx index 463de1f6223..89395aeee10 100644 --- a/app/src/organisms/GripperWizardFlows/BeforeBeginning.tsx +++ b/app/src/organisms/GripperWizardFlows/BeforeBeginning.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useEffect } from 'react' import { Trans, useTranslation } from 'react-i18next' import { COLORS, LegacyStyledText } from '@opentrons/components' import { EXTENSION } from '@opentrons/shared-data' @@ -76,7 +76,7 @@ export const BeforeBeginning = ( createdMaintenanceRunId, } = props const { t } = useTranslation(['gripper_wizard_flows', 'shared', 'branded']) - React.useEffect(() => { + useEffect(() => { if (createdMaintenanceRunId == null) { createMaintenanceRun({}) } diff --git a/app/src/organisms/GripperWizardFlows/ExitConfirmation.tsx b/app/src/organisms/GripperWizardFlows/ExitConfirmation.tsx index 3994536df78..28fb21635ac 100644 --- a/app/src/organisms/GripperWizardFlows/ExitConfirmation.tsx +++ b/app/src/organisms/GripperWizardFlows/ExitConfirmation.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/GripperWizardFlows/GripperWizardFlows.stories.tsx b/app/src/organisms/GripperWizardFlows/GripperWizardFlows.stories.tsx index b960c38ce77..b5318daf5d8 100644 --- a/app/src/organisms/GripperWizardFlows/GripperWizardFlows.stories.tsx +++ b/app/src/organisms/GripperWizardFlows/GripperWizardFlows.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { QueryClient, QueryClientProvider } from 'react-query' import { Provider } from 'react-redux' import { createStore } from 'redux' diff --git a/app/src/organisms/GripperWizardFlows/MountGripper.tsx b/app/src/organisms/GripperWizardFlows/MountGripper.tsx index acda62ef28c..4a428ac69b1 100644 --- a/app/src/organisms/GripperWizardFlows/MountGripper.tsx +++ b/app/src/organisms/GripperWizardFlows/MountGripper.tsx @@ -14,7 +14,7 @@ import { } from '@opentrons/components' import { useInstrumentsQuery } from '@opentrons/react-api-client' import { css } from 'styled-components' -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { getIsOnDevice } from '/app/redux/config' import { SmallButton } from '/app/atoms/buttons' @@ -62,8 +62,8 @@ export const MountGripper = ( const { proceed, isRobotMoving } = props const { t } = useTranslation(['gripper_wizard_flows', 'shared', 'branded']) const isOnDevice = useSelector(getIsOnDevice) - const [showUnableToDetect, setShowUnableToDetect] = React.useState(false) - const [isPending, setIsPending] = React.useState(false) + const [showUnableToDetect, setShowUnableToDetect] = useState(false) + const [isPending, setIsPending] = useState(false) const { data: instrumentsQueryData, refetch } = useInstrumentsQuery({ refetchInterval: QUICK_GRIPPER_POLL_MS, }) diff --git a/app/src/organisms/GripperWizardFlows/MovePin.tsx b/app/src/organisms/GripperWizardFlows/MovePin.tsx index 1f62f769daa..1cf5153eaa5 100644 --- a/app/src/organisms/GripperWizardFlows/MovePin.tsx +++ b/app/src/organisms/GripperWizardFlows/MovePin.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation, Trans } from 'react-i18next' import { EXTENSION } from '@opentrons/shared-data' import { diff --git a/app/src/organisms/GripperWizardFlows/Success.tsx b/app/src/organisms/GripperWizardFlows/Success.tsx index 9d4e4a0ec14..bbd36b4ca37 100644 --- a/app/src/organisms/GripperWizardFlows/Success.tsx +++ b/app/src/organisms/GripperWizardFlows/Success.tsx @@ -1,5 +1,4 @@ import { useSelector } from 'react-redux' -import * as React from 'react' import { css } from 'styled-components' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/GripperWizardFlows/UnmountGripper.tsx b/app/src/organisms/GripperWizardFlows/UnmountGripper.tsx index f6a36e1cfed..273b211ec4d 100644 --- a/app/src/organisms/GripperWizardFlows/UnmountGripper.tsx +++ b/app/src/organisms/GripperWizardFlows/UnmountGripper.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' import { @@ -55,7 +55,7 @@ export const UnmountGripper = ( const { proceed, isRobotMoving, goBack, chainRunCommands } = props const { t } = useTranslation(['gripper_wizard_flows', 'shared', 'branded']) const isOnDevice = useSelector(getIsOnDevice) - const [isPending, setIsPending] = React.useState(false) + const [isPending, setIsPending] = useState(false) const { data: instrumentsQueryData, refetch } = useInstrumentsQuery({ refetchInterval: QUICK_GRIPPER_POLL_MS, }) @@ -63,10 +63,9 @@ export const UnmountGripper = ( (i): i is GripperData => i.instrumentType === 'gripper' && i.ok ) - const [ - showGripperStillDetected, - setShowGripperStillDetected, - ] = React.useState(false) + const [showGripperStillDetected, setShowGripperStillDetected] = useState( + false + ) const handleContinue = (): void => { setIsPending(true) refetch() diff --git a/app/src/organisms/GripperWizardFlows/__tests__/BeforeBeginning.test.tsx b/app/src/organisms/GripperWizardFlows/__tests__/BeforeBeginning.test.tsx index d2a2fda6b64..888e6ba30b2 100644 --- a/app/src/organisms/GripperWizardFlows/__tests__/BeforeBeginning.test.tsx +++ b/app/src/organisms/GripperWizardFlows/__tests__/BeforeBeginning.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen, waitFor } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/GripperWizardFlows/__tests__/ExitConfirmation.test.tsx b/app/src/organisms/GripperWizardFlows/__tests__/ExitConfirmation.test.tsx index 9865b9f3143..4b20f234bfb 100644 --- a/app/src/organisms/GripperWizardFlows/__tests__/ExitConfirmation.test.tsx +++ b/app/src/organisms/GripperWizardFlows/__tests__/ExitConfirmation.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, expect } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/GripperWizardFlows/__tests__/MountGripper.test.tsx b/app/src/organisms/GripperWizardFlows/__tests__/MountGripper.test.tsx index 6a351304f78..cb2020c5bfe 100644 --- a/app/src/organisms/GripperWizardFlows/__tests__/MountGripper.test.tsx +++ b/app/src/organisms/GripperWizardFlows/__tests__/MountGripper.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen, waitFor } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/GripperWizardFlows/__tests__/MovePin.test.tsx b/app/src/organisms/GripperWizardFlows/__tests__/MovePin.test.tsx index 136d7f01da1..ecf3ef6f3c0 100644 --- a/app/src/organisms/GripperWizardFlows/__tests__/MovePin.test.tsx +++ b/app/src/organisms/GripperWizardFlows/__tests__/MovePin.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect, afterEach } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/GripperWizardFlows/__tests__/Success.test.tsx b/app/src/organisms/GripperWizardFlows/__tests__/Success.test.tsx index ad42aca7da3..d071fda3c69 100644 --- a/app/src/organisms/GripperWizardFlows/__tests__/Success.test.tsx +++ b/app/src/organisms/GripperWizardFlows/__tests__/Success.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, expect } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/GripperWizardFlows/__tests__/UnmountGripper.test.tsx b/app/src/organisms/GripperWizardFlows/__tests__/UnmountGripper.test.tsx index 64ddbe1e709..73c2d863366 100644 --- a/app/src/organisms/GripperWizardFlows/__tests__/UnmountGripper.test.tsx +++ b/app/src/organisms/GripperWizardFlows/__tests__/UnmountGripper.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, beforeEach, expect } from 'vitest' import { fireEvent, screen, waitFor } from '@testing-library/react' diff --git a/app/src/organisms/GripperWizardFlows/index.tsx b/app/src/organisms/GripperWizardFlows/index.tsx index 553c3a7308a..ceb19eecbf3 100644 --- a/app/src/organisms/GripperWizardFlows/index.tsx +++ b/app/src/organisms/GripperWizardFlows/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' @@ -68,7 +68,7 @@ export function GripperWizardFlows( isLoading: isCommandLoading, } = useCreateMaintenanceCommandMutation() - const [createdMaintenanceRunId, setCreatedMaintenanceRunId] = React.useState< + const [createdMaintenanceRunId, setCreatedMaintenanceRunId] = useState< string | null >(null) @@ -77,7 +77,7 @@ export function GripperWizardFlows( const [ monitorMaintenanceRunForDeletion, setMonitorMaintenanceRunForDeletion, - ] = React.useState(false) + ] = useState(false) const { createTargetedMaintenanceRun, @@ -95,7 +95,7 @@ export function GripperWizardFlows( // this will close the modal in case the run was deleted by the terminate // activity modal on the ODD - React.useEffect(() => { + useEffect(() => { if ( createdMaintenanceRunId !== null && maintenanceRunData?.data.id === createdMaintenanceRunId @@ -116,16 +116,17 @@ export function GripperWizardFlows( closeFlow, ]) - const [isExiting, setIsExiting] = React.useState(false) - const [errorMessage, setErrorMessage] = React.useState(null) + const [isExiting, setIsExiting] = useState(false) + const [errorMessage, setErrorMessage] = useState(null) const handleClose = (): void => { - if (props?.onComplete != null) props.onComplete() + if (props?.onComplete != null) { + props.onComplete() + } if (maintenanceRunData != null) { deleteMaintenanceRun(maintenanceRunData?.data.id) - } else { - closeFlow() } + closeFlow() } const { @@ -141,20 +142,21 @@ export function GripperWizardFlows( }) const handleCleanUpAndClose = (): void => { - if (maintenanceRunData?.data.id == null) handleClose() - else { + if (maintenanceRunData?.data.id == null) { + handleClose() + } else { chainRunCommands( maintenanceRunData?.data.id, [{ commandType: 'home' as const, params: {} }], false ) - .then(() => { - handleClose() - }) .catch(error => { setIsExiting(true) setErrorMessage(error.message as string) }) + .finally(() => { + handleClose() + }) } } @@ -234,11 +236,8 @@ export const GripperWizard = ( const isOnDevice = useSelector(getIsOnDevice) const { t } = useTranslation('gripper_wizard_flows') const gripperWizardSteps = getGripperWizardSteps(flowType) - const [currentStepIndex, setCurrentStepIndex] = React.useState(0) - const [ - frontJawOffset, - setFrontJawOffset, - ] = React.useState(null) + const [currentStepIndex, setCurrentStepIndex] = useState(0) + const [frontJawOffset, setFrontJawOffset] = useState(null) const totalStepCount = gripperWizardSteps.length - 1 const currentStep = gripperWizardSteps?.[currentStepIndex] diff --git a/app/src/organisms/HowCalibrationWorksModal/__tests__/HowCalibrationWorksModal.test.tsx b/app/src/organisms/HowCalibrationWorksModal/__tests__/HowCalibrationWorksModal.test.tsx index 329d3ed3e5f..8eee1543c5a 100644 --- a/app/src/organisms/HowCalibrationWorksModal/__tests__/HowCalibrationWorksModal.test.tsx +++ b/app/src/organisms/HowCalibrationWorksModal/__tests__/HowCalibrationWorksModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import '@testing-library/jest-dom/vitest' import { describe, it, vi, beforeEach, expect } from 'vitest' diff --git a/app/src/organisms/HowCalibrationWorksModal/index.tsx b/app/src/organisms/HowCalibrationWorksModal/index.tsx index e0f4bda2972..525b94cb636 100644 --- a/app/src/organisms/HowCalibrationWorksModal/index.tsx +++ b/app/src/organisms/HowCalibrationWorksModal/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' diff --git a/app/src/organisms/IncompatibleModule/IncompatibleModuleDesktopModalBody.tsx b/app/src/organisms/IncompatibleModule/IncompatibleModuleDesktopModalBody.tsx index 036c0db566f..a1fd7087b06 100644 --- a/app/src/organisms/IncompatibleModule/IncompatibleModuleDesktopModalBody.tsx +++ b/app/src/organisms/IncompatibleModule/IncompatibleModuleDesktopModalBody.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation, Trans } from 'react-i18next' import { DIRECTION_COLUMN, diff --git a/app/src/organisms/IncompatibleModule/IncompatibleModuleODDModalBody.tsx b/app/src/organisms/IncompatibleModule/IncompatibleModuleODDModalBody.tsx index fb08e1a8bcc..459871b4803 100644 --- a/app/src/organisms/IncompatibleModule/IncompatibleModuleODDModalBody.tsx +++ b/app/src/organisms/IncompatibleModule/IncompatibleModuleODDModalBody.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation, Trans } from 'react-i18next' import capitalize from 'lodash/capitalize' import { diff --git a/app/src/organisms/IncompatibleModule/IncompatibleModuleTakeover.tsx b/app/src/organisms/IncompatibleModule/IncompatibleModuleTakeover.tsx index 3be98ad14fc..52e75a0aead 100644 --- a/app/src/organisms/IncompatibleModule/IncompatibleModuleTakeover.tsx +++ b/app/src/organisms/IncompatibleModule/IncompatibleModuleTakeover.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { createPortal } from 'react-dom' import { IncompatibleModuleODDModalBody } from './IncompatibleModuleODDModalBody' import { IncompatibleModuleDesktopModalBody } from './IncompatibleModuleDesktopModalBody' diff --git a/app/src/organisms/IncompatibleModule/__tests__/IncompatibleModuleDesktopModalBody.test.tsx b/app/src/organisms/IncompatibleModule/__tests__/IncompatibleModuleDesktopModalBody.test.tsx index 245073c15a6..41edc4439c6 100644 --- a/app/src/organisms/IncompatibleModule/__tests__/IncompatibleModuleDesktopModalBody.test.tsx +++ b/app/src/organisms/IncompatibleModule/__tests__/IncompatibleModuleDesktopModalBody.test.tsx @@ -1,4 +1,4 @@ -import React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, beforeEach, vi } from 'vitest' import { when } from 'vitest-when' diff --git a/app/src/organisms/IncompatibleModule/__tests__/IncompatibleModuleODDModalBody.test.tsx b/app/src/organisms/IncompatibleModule/__tests__/IncompatibleModuleODDModalBody.test.tsx index 1a6ce6854ae..e2d4e1a2af0 100644 --- a/app/src/organisms/IncompatibleModule/__tests__/IncompatibleModuleODDModalBody.test.tsx +++ b/app/src/organisms/IncompatibleModule/__tests__/IncompatibleModuleODDModalBody.test.tsx @@ -1,4 +1,4 @@ -import React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, beforeEach, expect } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/IncompatibleModule/__tests__/IncompatibleModuleTakeover.test.tsx b/app/src/organisms/IncompatibleModule/__tests__/IncompatibleModuleTakeover.test.tsx index b8615e62d36..0a7f6b05db2 100644 --- a/app/src/organisms/IncompatibleModule/__tests__/IncompatibleModuleTakeover.test.tsx +++ b/app/src/organisms/IncompatibleModule/__tests__/IncompatibleModuleTakeover.test.tsx @@ -1,4 +1,4 @@ -import React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, beforeEach, afterEach, expect, vi } from 'vitest' import { when } from 'vitest-when' diff --git a/app/src/organisms/IncompatibleModule/hooks/__tests__/useIncompatibleModulesAttached.test.tsx b/app/src/organisms/IncompatibleModule/hooks/__tests__/useIncompatibleModulesAttached.test.tsx index 970805c95c6..7e1a7342db1 100644 --- a/app/src/organisms/IncompatibleModule/hooks/__tests__/useIncompatibleModulesAttached.test.tsx +++ b/app/src/organisms/IncompatibleModule/hooks/__tests__/useIncompatibleModulesAttached.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { QueryClient, QueryClientProvider } from 'react-query' import { vi, it, expect, describe, beforeEach } from 'vitest' diff --git a/app/src/organisms/InstrumentInfo/__tests__/InstrumentInfo.test.tsx b/app/src/organisms/InstrumentInfo/__tests__/InstrumentInfo.test.tsx index ac6387cfc24..4b857d4d288 100644 --- a/app/src/organisms/InstrumentInfo/__tests__/InstrumentInfo.test.tsx +++ b/app/src/organisms/InstrumentInfo/__tests__/InstrumentInfo.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/InstrumentMountItem/AttachedInstrumentMountItem.tsx b/app/src/organisms/InstrumentMountItem/AttachedInstrumentMountItem.tsx index d83e3d4e8fd..7d54755f86e 100644 --- a/app/src/organisms/InstrumentMountItem/AttachedInstrumentMountItem.tsx +++ b/app/src/organisms/InstrumentMountItem/AttachedInstrumentMountItem.tsx @@ -6,7 +6,7 @@ import { SINGLE_MOUNT_PIPETTES } from '@opentrons/shared-data' import { useGripperDisplayName, usePipetteModelSpecs, -} from '/app/resources/instruments/hooks' +} from '/app/local-resources/instruments' import { ChoosePipette } from '../PipetteWizardFlows/ChoosePipette' import { FLOWS } from '../PipetteWizardFlows/constants' import { GRIPPER_FLOW_TYPES } from '../GripperWizardFlows/constants' diff --git a/app/src/organisms/InstrumentMountItem/LabeledMount.tsx b/app/src/organisms/InstrumentMountItem/LabeledMount.tsx index 668d7b960a0..20fe3941604 100644 --- a/app/src/organisms/InstrumentMountItem/LabeledMount.tsx +++ b/app/src/organisms/InstrumentMountItem/LabeledMount.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import styled from 'styled-components' import { diff --git a/app/src/organisms/InstrumentMountItem/ProtocolInstrumentMountItem.tsx b/app/src/organisms/InstrumentMountItem/ProtocolInstrumentMountItem.tsx index 459b584f882..3ae2b2e85d2 100644 --- a/app/src/organisms/InstrumentMountItem/ProtocolInstrumentMountItem.tsx +++ b/app/src/organisms/InstrumentMountItem/ProtocolInstrumentMountItem.tsx @@ -22,7 +22,7 @@ import { SmallButton } from '/app/atoms/buttons' import { useGripperDisplayName, usePipetteNameSpecs, -} from '/app/resources/instruments/hooks' +} from '/app/local-resources/instruments' import { FLOWS } from '../PipetteWizardFlows/constants' import { PipetteWizardFlows } from '../PipetteWizardFlows' import { GripperWizardFlows } from '../GripperWizardFlows' diff --git a/app/src/organisms/InstrumentMountItem/__tests__/ProtocolInstrumentMountItem.test.tsx b/app/src/organisms/InstrumentMountItem/__tests__/ProtocolInstrumentMountItem.test.tsx index eb4aa97ca65..3ed34b5a7d6 100644 --- a/app/src/organisms/InstrumentMountItem/__tests__/ProtocolInstrumentMountItem.test.tsx +++ b/app/src/organisms/InstrumentMountItem/__tests__/ProtocolInstrumentMountItem.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, beforeEach } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' import { LEFT } from '@opentrons/shared-data' diff --git a/app/src/organisms/InterventionModal/InterventionCommandMessage.tsx b/app/src/organisms/InterventionModal/InterventionCommandMessage.tsx index 97fc9e7f69c..d9770e4a621 100644 --- a/app/src/organisms/InterventionModal/InterventionCommandMessage.tsx +++ b/app/src/organisms/InterventionModal/InterventionCommandMessage.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' diff --git a/app/src/organisms/InterventionModal/InterventionModal.stories.tsx b/app/src/organisms/InterventionModal/InterventionModal.stories.tsx index f9f13bb583e..ef378f49f70 100644 --- a/app/src/organisms/InterventionModal/InterventionModal.stories.tsx +++ b/app/src/organisms/InterventionModal/InterventionModal.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Provider } from 'react-redux' import { createStore } from 'redux' import { QueryClient, QueryClientProvider } from 'react-query' diff --git a/app/src/organisms/InterventionModal/LabwareDisabledOverlay.tsx b/app/src/organisms/InterventionModal/LabwareDisabledOverlay.tsx index f301571f4f5..30c742c42fe 100644 --- a/app/src/organisms/InterventionModal/LabwareDisabledOverlay.tsx +++ b/app/src/organisms/InterventionModal/LabwareDisabledOverlay.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - import { COLORS } from '@opentrons/components' import type { LabwareDefinition2 } from '@opentrons/shared-data' diff --git a/app/src/organisms/InterventionModal/MoveLabwareInterventionContent.tsx b/app/src/organisms/InterventionModal/MoveLabwareInterventionContent.tsx index 29049bd82a0..d44a96ecfa8 100644 --- a/app/src/organisms/InterventionModal/MoveLabwareInterventionContent.tsx +++ b/app/src/organisms/InterventionModal/MoveLabwareInterventionContent.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' diff --git a/app/src/organisms/InterventionModal/PauseInterventionContent.tsx b/app/src/organisms/InterventionModal/PauseInterventionContent.tsx index 36f313873f6..72f09e522b6 100644 --- a/app/src/organisms/InterventionModal/PauseInterventionContent.tsx +++ b/app/src/organisms/InterventionModal/PauseInterventionContent.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' @@ -81,7 +81,7 @@ interface PauseHeaderProps { function PauseHeader({ startedAt }: PauseHeaderProps): JSX.Element { const { t, i18n } = useTranslation('run_details') - const [now, setNow] = React.useState(Date()) + const [now, setNow] = useState(Date()) useInterval( () => { setNow(Date()) diff --git a/app/src/organisms/InterventionModal/__tests__/InterventionCommandMesage.test.tsx b/app/src/organisms/InterventionModal/__tests__/InterventionCommandMesage.test.tsx index 48f2508435f..6f3a688b808 100644 --- a/app/src/organisms/InterventionModal/__tests__/InterventionCommandMesage.test.tsx +++ b/app/src/organisms/InterventionModal/__tests__/InterventionCommandMesage.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, expect } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/InterventionModal/__tests__/InterventionCommandMessage.test.tsx b/app/src/organisms/InterventionModal/__tests__/InterventionCommandMessage.test.tsx index 48f2508435f..6f3a688b808 100644 --- a/app/src/organisms/InterventionModal/__tests__/InterventionCommandMessage.test.tsx +++ b/app/src/organisms/InterventionModal/__tests__/InterventionCommandMessage.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, expect } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/InterventionModal/__tests__/InterventionModal.test.tsx b/app/src/organisms/InterventionModal/__tests__/InterventionModal.test.tsx index 2a6acb3be85..b343d316eef 100644 --- a/app/src/organisms/InterventionModal/__tests__/InterventionModal.test.tsx +++ b/app/src/organisms/InterventionModal/__tests__/InterventionModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, renderHook, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/InterventionModal/__tests__/LabwareDisabledOverlay.test.tsx b/app/src/organisms/InterventionModal/__tests__/LabwareDisabledOverlay.test.tsx index b217edb116d..d6b8760f368 100644 --- a/app/src/organisms/InterventionModal/__tests__/LabwareDisabledOverlay.test.tsx +++ b/app/src/organisms/InterventionModal/__tests__/LabwareDisabledOverlay.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { render, screen } from '@testing-library/react' import { describe, it, expect } from 'vitest' import { COLORS } from '@opentrons/components' diff --git a/app/src/organisms/InterventionModal/index.tsx b/app/src/organisms/InterventionModal/index.tsx index dff08559f9d..c8372a83d91 100644 --- a/app/src/organisms/InterventionModal/index.tsx +++ b/app/src/organisms/InterventionModal/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' @@ -112,7 +111,8 @@ export function InterventionModal({ const isOnDevice = useSelector(getIsOnDevice) const robotType = useRobotType(robotName) - const childContent = React.useMemo(() => { + // TODO(jh 09-19-24): Make this into its own component. + const childContent = (() => { switch (command.commandType) { case 'waitForResume': case 'pause': // legacy pause command @@ -136,12 +136,7 @@ export function InterventionModal({ ) return null } - }, [ - command.id, - analysis?.status, - run.labware.map(l => l.id).join(), - run.modules.map(m => m.id).join(), - ]) + })() const { iconName, headerTitle, headerTitleOnDevice } = (() => { switch (command.commandType) { diff --git a/app/src/organisms/LabwareOffsetTabs/__tests__/LabwareOffsetTabs.test.tsx b/app/src/organisms/LabwareOffsetTabs/__tests__/LabwareOffsetTabs.test.tsx index 3029cecdd2f..d5d2a89538a 100644 --- a/app/src/organisms/LabwareOffsetTabs/__tests__/LabwareOffsetTabs.test.tsx +++ b/app/src/organisms/LabwareOffsetTabs/__tests__/LabwareOffsetTabs.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/LabwareOffsetTabs/index.tsx b/app/src/organisms/LabwareOffsetTabs/index.tsx index 36dd745f682..3ad81c51d01 100644 --- a/app/src/organisms/LabwareOffsetTabs/index.tsx +++ b/app/src/organisms/LabwareOffsetTabs/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { BORDERS, @@ -27,7 +27,7 @@ export function LabwareOffsetTabs({ ...styleProps }: LabwareOffsetTabsProps): JSX.Element { const { t } = useTranslation('labware_position_check') - const [currentTab, setCurrentTab] = React.useState('table') + const [currentTab, setCurrentTab] = useState('table') const activeTabComponent = { table: TableComponent, diff --git a/app/src/organisms/LabwarePositionCheck/ExitConfirmation.tsx b/app/src/organisms/LabwarePositionCheck/ExitConfirmation.tsx index 47cb0c11cab..ab6cb857035 100644 --- a/app/src/organisms/LabwarePositionCheck/ExitConfirmation.tsx +++ b/app/src/organisms/LabwarePositionCheck/ExitConfirmation.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import styled from 'styled-components' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/LabwarePositionCheck/FatalErrorModal.tsx b/app/src/organisms/LabwarePositionCheck/FatalErrorModal.tsx index 96e6f920d47..69d9d6f1028 100644 --- a/app/src/organisms/LabwarePositionCheck/FatalErrorModal.tsx +++ b/app/src/organisms/LabwarePositionCheck/FatalErrorModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { createPortal } from 'react-dom' import styled from 'styled-components' import { useTranslation } from 'react-i18next' diff --git a/app/src/organisms/LabwarePositionCheck/IntroScreen/index.tsx b/app/src/organisms/LabwarePositionCheck/IntroScreen/index.tsx index 75fc6b8a62e..aa7dd002c2a 100644 --- a/app/src/organisms/LabwarePositionCheck/IntroScreen/index.tsx +++ b/app/src/organisms/LabwarePositionCheck/IntroScreen/index.tsx @@ -21,7 +21,7 @@ import { import { RobotMotionLoader } from '../RobotMotionLoader' import { getPrepCommands } from './getPrepCommands' import { WizardRequiredEquipmentList } from '/app/molecules/WizardRequiredEquipmentList' -import { getLatestCurrentOffsets } from '../../Devices/ProtocolRun/SetupLabwarePositionCheck/utils' +import { getLatestCurrentOffsets } from '/app/transformations/runs' import { getIsOnDevice } from '/app/redux/config' import { NeedHelpLink } from '../../CalibrationPanels' import { useSelector } from 'react-redux' diff --git a/app/src/organisms/LabwarePositionCheck/LabwarePositionCheckComponent.tsx b/app/src/organisms/LabwarePositionCheck/LabwarePositionCheckComponent.tsx index 3174ef5f9de..2671d98b611 100644 --- a/app/src/organisms/LabwarePositionCheck/LabwarePositionCheckComponent.tsx +++ b/app/src/organisms/LabwarePositionCheck/LabwarePositionCheckComponent.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect, useReducer } from 'react' import { createPortal } from 'react-dom' import isEqual from 'lodash/isEqual' import { useSelector } from 'react-redux' @@ -85,7 +85,7 @@ export const LabwarePositionCheckComponent = ( const [ monitorMaintenanceRunForDeletion, setMonitorMaintenanceRunForDeletion, - ] = React.useState(false) + ] = useState(false) const { data: maintenanceRunData } = useNotifyCurrentMaintenanceRun({ refetchInterval: RUN_REFETCH_INTERVAL, @@ -94,7 +94,7 @@ export const LabwarePositionCheckComponent = ( // this will close the modal in case the run was deleted by the terminate // activity modal on the ODD - React.useEffect(() => { + useEffect(() => { if ( maintenanceRunId !== null && maintenanceRunData?.data.id === maintenanceRunId @@ -114,14 +114,9 @@ export const LabwarePositionCheckComponent = ( setMaintenanceRunId, ]) - const [fatalError, setFatalError] = React.useState(null) - const [isApplyingOffsets, setIsApplyingOffsets] = React.useState( - false - ) - const [ - { workingOffsets, tipPickUpOffset }, - registerPosition, - ] = React.useReducer( + const [fatalError, setFatalError] = useState(null) + const [isApplyingOffsets, setIsApplyingOffsets] = useState(false) + const [{ workingOffsets, tipPickUpOffset }, registerPosition] = useReducer( ( state: { workingOffsets: WorkingOffset[] @@ -189,7 +184,7 @@ export const LabwarePositionCheckComponent = ( }, { workingOffsets: [], tipPickUpOffset: null } ) - const [isExiting, setIsExiting] = React.useState(false) + const [isExiting, setIsExiting] = useState(false) const { createMaintenanceCommand: createSilentCommand, } = useCreateMaintenanceCommandMutation() @@ -199,7 +194,7 @@ export const LabwarePositionCheckComponent = ( } = useChainMaintenanceCommands() const { createLabwareOffset } = useCreateLabwareOffsetMutation() - const [currentStepIndex, setCurrentStepIndex] = React.useState(0) + const [currentStepIndex, setCurrentStepIndex] = useState(0) const handleCleanUpAndClose = (): void => { setIsExiting(true) const dropTipToBeSafeCommands: DropTipCreateCommand[] = shouldUseMetalProbe diff --git a/app/src/organisms/LabwarePositionCheck/LiveOffsetValue.tsx b/app/src/organisms/LabwarePositionCheck/LiveOffsetValue.tsx index 294755a73ca..21b3af917f9 100644 --- a/app/src/organisms/LabwarePositionCheck/LiveOffsetValue.tsx +++ b/app/src/organisms/LabwarePositionCheck/LiveOffsetValue.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { Fragment } from 'react' import { ALIGN_CENTER, BORDERS, @@ -55,7 +55,7 @@ export function LiveOffsetValue(props: OffsetVectorProps): JSX.Element { > {[x, y, z].map((axis, index) => ( - + {axis.toFixed(1)} - + ))} diff --git a/app/src/organisms/LabwarePositionCheck/PrepareSpace.tsx b/app/src/organisms/LabwarePositionCheck/PrepareSpace.tsx index 12e630b6f00..c26af819b10 100644 --- a/app/src/organisms/LabwarePositionCheck/PrepareSpace.tsx +++ b/app/src/organisms/LabwarePositionCheck/PrepareSpace.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import styled, { css } from 'styled-components' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' diff --git a/app/src/organisms/LabwarePositionCheck/ResultsSummary.tsx b/app/src/organisms/LabwarePositionCheck/ResultsSummary.tsx index bbf9fd55e38..8fdd84884f7 100644 --- a/app/src/organisms/LabwarePositionCheck/ResultsSummary.tsx +++ b/app/src/organisms/LabwarePositionCheck/ResultsSummary.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useMemo, Fragment } from 'react' import styled, { css } from 'styled-components' import { useSelector } from 'react-redux' import isEqual from 'lodash/isEqual' @@ -84,7 +84,7 @@ export const ResultsSummary = ( ) const isOnDevice = useSelector(getIsOnDevice) - const offsetsToApply = React.useMemo(() => { + const offsetsToApply = useMemo(() => { return workingOffsets.map( ({ initialPosition, finalPosition, labwareId, location }) => { const definitionUri = @@ -321,7 +321,7 @@ const OffsetTable = (props: OffsetTableProps): JSX.Element => { ) : ( {[vector.x, vector.y, vector.z].map((axis, index) => ( - + 0 ? SPACING.spacing8 : 0} @@ -333,7 +333,7 @@ const OffsetTable = (props: OffsetTableProps): JSX.Element => { {axis.toFixed(1)} - + ))} )} @@ -398,7 +398,7 @@ export const TerseOffsetTable = (props: OffsetTableProps): JSX.Element => { ) : ( {[vector.x, vector.y, vector.z].map((axis, index) => ( - + { > {axis.toFixed(1)} - + ))} )} diff --git a/app/src/organisms/LabwarePositionCheck/ReturnTip.tsx b/app/src/organisms/LabwarePositionCheck/ReturnTip.tsx index fe01316894d..8ba2f78d125 100644 --- a/app/src/organisms/LabwarePositionCheck/ReturnTip.tsx +++ b/app/src/organisms/LabwarePositionCheck/ReturnTip.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { DIRECTION_COLUMN, diff --git a/app/src/organisms/LabwarePositionCheck/RobotMotionLoader.tsx b/app/src/organisms/LabwarePositionCheck/RobotMotionLoader.tsx index dbb5f6973f2..577dae6eff7 100644 --- a/app/src/organisms/LabwarePositionCheck/RobotMotionLoader.tsx +++ b/app/src/organisms/LabwarePositionCheck/RobotMotionLoader.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import styled from 'styled-components' import { ALIGN_CENTER, diff --git a/app/src/organisms/LabwarePositionCheck/TerseOffsetTable.stories.tsx b/app/src/organisms/LabwarePositionCheck/TerseOffsetTable.stories.tsx index 3e324412002..da7c52de513 100644 --- a/app/src/organisms/LabwarePositionCheck/TerseOffsetTable.stories.tsx +++ b/app/src/organisms/LabwarePositionCheck/TerseOffsetTable.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { ALIGN_FLEX_END, DIRECTION_COLUMN, diff --git a/app/src/organisms/LabwarePositionCheck/TipConfirmation.tsx b/app/src/organisms/LabwarePositionCheck/TipConfirmation.tsx index 35cdc3230e0..9a4df89265f 100644 --- a/app/src/organisms/LabwarePositionCheck/TipConfirmation.tsx +++ b/app/src/organisms/LabwarePositionCheck/TipConfirmation.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { ALIGN_CENTER, COLORS, diff --git a/app/src/organisms/LabwarePositionCheck/TwoUpTileLayout.tsx b/app/src/organisms/LabwarePositionCheck/TwoUpTileLayout.tsx index 44ee775f25a..7c6cd309bb4 100644 --- a/app/src/organisms/LabwarePositionCheck/TwoUpTileLayout.tsx +++ b/app/src/organisms/LabwarePositionCheck/TwoUpTileLayout.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import styled, { css } from 'styled-components' import { DIRECTION_COLUMN, diff --git a/app/src/organisms/LabwarePositionCheck/__tests__/CheckItem.test.tsx b/app/src/organisms/LabwarePositionCheck/__tests__/CheckItem.test.tsx index cf0b61a8036..82fb4da874a 100644 --- a/app/src/organisms/LabwarePositionCheck/__tests__/CheckItem.test.tsx +++ b/app/src/organisms/LabwarePositionCheck/__tests__/CheckItem.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, afterEach, expect } from 'vitest' diff --git a/app/src/organisms/LabwarePositionCheck/__tests__/ExitConfirmation.test.tsx b/app/src/organisms/LabwarePositionCheck/__tests__/ExitConfirmation.test.tsx index 2fd71e3acf0..6a93da71dc5 100644 --- a/app/src/organisms/LabwarePositionCheck/__tests__/ExitConfirmation.test.tsx +++ b/app/src/organisms/LabwarePositionCheck/__tests__/ExitConfirmation.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach, afterEach, expect, vi } from 'vitest' import { ExitConfirmation } from '../ExitConfirmation' diff --git a/app/src/organisms/LabwarePositionCheck/__tests__/PickUpTip.test.tsx b/app/src/organisms/LabwarePositionCheck/__tests__/PickUpTip.test.tsx index c852c5eee5a..c23e1c1af2c 100644 --- a/app/src/organisms/LabwarePositionCheck/__tests__/PickUpTip.test.tsx +++ b/app/src/organisms/LabwarePositionCheck/__tests__/PickUpTip.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen, waitFor } from '@testing-library/react' import { it, describe, beforeEach, vi, afterEach, expect } from 'vitest' import { FLEX_ROBOT_TYPE, HEATERSHAKER_MODULE_V1 } from '@opentrons/shared-data' diff --git a/app/src/organisms/LabwarePositionCheck/__tests__/ResultsSummary.test.tsx b/app/src/organisms/LabwarePositionCheck/__tests__/ResultsSummary.test.tsx index e66d900e76e..24101904de4 100644 --- a/app/src/organisms/LabwarePositionCheck/__tests__/ResultsSummary.test.tsx +++ b/app/src/organisms/LabwarePositionCheck/__tests__/ResultsSummary.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, beforeEach, afterEach, expect, vi } from 'vitest' import { fireEvent, screen } from '@testing-library/react' import { i18n } from '/app/i18n' diff --git a/app/src/organisms/LabwarePositionCheck/__tests__/ReturnTip.test.tsx b/app/src/organisms/LabwarePositionCheck/__tests__/ReturnTip.test.tsx index 0c44d4ecf89..0af86097f9c 100644 --- a/app/src/organisms/LabwarePositionCheck/__tests__/ReturnTip.test.tsx +++ b/app/src/organisms/LabwarePositionCheck/__tests__/ReturnTip.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest' diff --git a/app/src/organisms/LabwarePositionCheck/__tests__/RobotMotionLoader.test.tsx b/app/src/organisms/LabwarePositionCheck/__tests__/RobotMotionLoader.test.tsx index f12c52b07d4..fc2c49aa8f5 100644 --- a/app/src/organisms/LabwarePositionCheck/__tests__/RobotMotionLoader.test.tsx +++ b/app/src/organisms/LabwarePositionCheck/__tests__/RobotMotionLoader.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { describe, it } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/LabwarePositionCheck/__tests__/TipConfirmation.test.tsx b/app/src/organisms/LabwarePositionCheck/__tests__/TipConfirmation.test.tsx index 3217cea0dbe..8f8878a7122 100644 --- a/app/src/organisms/LabwarePositionCheck/__tests__/TipConfirmation.test.tsx +++ b/app/src/organisms/LabwarePositionCheck/__tests__/TipConfirmation.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, beforeEach, afterEach, expect, vi } from 'vitest' import { fireEvent, screen } from '@testing-library/react' import { TipConfirmation } from '../TipConfirmation' diff --git a/app/src/organisms/LabwarePositionCheck/__tests__/useLaunchLPC.test.tsx b/app/src/organisms/LabwarePositionCheck/__tests__/useLaunchLPC.test.tsx index 4c007581c81..fb983097d01 100644 --- a/app/src/organisms/LabwarePositionCheck/__tests__/useLaunchLPC.test.tsx +++ b/app/src/organisms/LabwarePositionCheck/__tests__/useLaunchLPC.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Provider } from 'react-redux' import configureStore from 'redux-mock-store' import { when } from 'vitest-when' @@ -22,8 +22,8 @@ import { renderWithProviders } from '/app/__testing-utils__' import { useCreateTargetedMaintenanceRunMutation, useNotifyRunQuery, + useMostRecentCompletedAnalysis, } from '/app/resources/runs' -import { useMostRecentCompletedAnalysis } from '../useMostRecentCompletedAnalysis' import { useLaunchLPC } from '../useLaunchLPC' import { LabwarePositionCheck } from '..' @@ -33,7 +33,6 @@ import type { LabwareDefinition2 } from '@opentrons/shared-data' vi.mock('../') vi.mock('@opentrons/react-api-client') -vi.mock('../useMostRecentCompletedAnalysis') vi.mock('/app/resources/runs') const MOCK_RUN_ID = 'mockRunId' diff --git a/app/src/organisms/LabwarePositionCheck/useLaunchLPC.tsx b/app/src/organisms/LabwarePositionCheck/useLaunchLPC.tsx index 23569a776fd..c3262af8225 100644 --- a/app/src/organisms/LabwarePositionCheck/useLaunchLPC.tsx +++ b/app/src/organisms/LabwarePositionCheck/useLaunchLPC.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useCreateMaintenanceRunLabwareDefinitionMutation, @@ -8,9 +8,9 @@ import { import { useCreateTargetedMaintenanceRunMutation, useNotifyRunQuery, + useMostRecentCompletedAnalysis, } from '/app/resources/runs' import { LabwarePositionCheck } from '.' -import { useMostRecentCompletedAnalysis } from './useMostRecentCompletedAnalysis' import { getLabwareDefinitionsFromCommands } from '/app/molecules/Command/utils/getLabwareDefinitionsFromCommands' import type { RobotType } from '@opentrons/shared-data' @@ -29,9 +29,7 @@ export function useLaunchLPC( isLoading: isDeletingMaintenanceRun, } = useDeleteMaintenanceRunMutation() const mostRecentAnalysis = useMostRecentCompletedAnalysis(runId) - const [maintenanceRunId, setMaintenanceRunId] = React.useState( - null - ) + const [maintenanceRunId, setMaintenanceRunId] = useState(null) const currentOffsets = runRecord?.data?.labwareOffsets ?? [] const { createLabwareDefinition, diff --git a/app/src/organisms/LabwarePositionCheck/utils/labware.ts b/app/src/organisms/LabwarePositionCheck/utils/labware.ts index efa4336bc78..2fd03ccc0c0 100644 --- a/app/src/organisms/LabwarePositionCheck/utils/labware.ts +++ b/app/src/organisms/LabwarePositionCheck/utils/labware.ts @@ -4,7 +4,7 @@ import { getTiprackVolume, getLabwareDefURI, } from '@opentrons/shared-data' -import { getModuleInitialLoadInfo } from '../../Devices/ProtocolRun/utils/getModuleInitialLoadInfo' +import { getModuleInitialLoadInfo } from '/app/transformations/commands' import { getLabwareDefinitionsFromCommands } from '/app/molecules/Command/utils/getLabwareDefinitionsFromCommands' import type { CompletedProtocolAnalysis, diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/LiquidDetailCard.tsx b/app/src/organisms/LiquidsLabwareDetailsModal/LiquidDetailCard.tsx similarity index 98% rename from app/src/organisms/Devices/ProtocolRun/SetupLiquids/LiquidDetailCard.tsx rename to app/src/organisms/LiquidsLabwareDetailsModal/LiquidDetailCard.tsx index 7178a8b6404..cb1591dc72f 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/LiquidDetailCard.tsx +++ b/app/src/organisms/LiquidsLabwareDetailsModal/LiquidDetailCard.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useSelector } from 'react-redux' import { css } from 'styled-components' import { @@ -25,7 +25,7 @@ import { ANALYTICS_HIGHLIGHT_LIQUID_IN_DETAIL_MODAL, } from '/app/redux/analytics' import { getIsOnDevice } from '/app/redux/config' -import { getWellRangeForLiquidLabwarePair } from './utils' +import { getWellRangeForLiquidLabwarePair } from '/app/transformations/analysis' export const CARD_OUTLINE_BORDER_STYLE = css` border-style: ${BORDERS.styleSolid}; diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/LiquidsLabwareDetailsModal.tsx b/app/src/organisms/LiquidsLabwareDetailsModal/LiquidsLabwareDetailsModal.tsx similarity index 93% rename from app/src/organisms/Devices/ProtocolRun/SetupLiquids/LiquidsLabwareDetailsModal.tsx rename to app/src/organisms/LiquidsLabwareDetailsModal/LiquidsLabwareDetailsModal.tsx index 733d767e8cd..62b5e3cf435 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/LiquidsLabwareDetailsModal.tsx +++ b/app/src/organisms/LiquidsLabwareDetailsModal/LiquidsLabwareDetailsModal.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useRef, useState, useEffect } from 'react' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' import { css } from 'styled-components' @@ -22,16 +22,18 @@ import { import { OddModal } from '/app/molecules/OddModal' import { getIsOnDevice } from '/app/redux/config' -import { useMostRecentCompletedAnalysis } from '../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' -import { getLocationInfoNames } from '../utils/getLocationInfoNames' -import { getSlotLabwareDefinition } from '../utils/getSlotLabwareDefinition' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' +import { + getLocationInfoNames, + getSlotLabwareDefinition, +} from '/app/transformations/commands' import { LiquidDetailCard } from './LiquidDetailCard' import { getLiquidsByIdForLabware, getDisabledWellFillFromLabwareId, getWellGroupForLiquidId, getDisabledWellGroupForLiquidId, -} from './utils' +} from '/app/transformations/analysis' interface LiquidsLabwareDetailsModalProps { liquidId?: string @@ -46,7 +48,7 @@ export const LiquidsLabwareDetailsModal = ( const { liquidId, labwareId, runId, closeModal } = props const { t } = useTranslation('protocol_setup') const isOnDevice = useSelector(getIsOnDevice) - const currentLiquidRef = React.useRef(null) + const currentLiquidRef = useRef(null) const protocolData = useMostRecentCompletedAnalysis(runId) const commands = protocolData?.commands ?? [] const liquids = parseLiquidsInLoadOrder( @@ -63,7 +65,7 @@ export const LiquidsLabwareDetailsModal = ( const filteredLiquidsInLoadOrder = liquids.filter(liquid => { return Object.keys(labwareInfo).some(key => key === liquid.id) }) - const [selectedValue, setSelectedValue] = React.useState( + const [selectedValue, setSelectedValue] = useState( liquidId ?? filteredLiquidsInLoadOrder[0].id ) @@ -77,7 +79,7 @@ export const LiquidsLabwareDetailsModal = ( const scrollToCurrentItem = (): void => { currentLiquidRef.current?.scrollIntoView({ behavior: 'smooth' }) } - React.useEffect(() => { + useEffect(() => { scrollToCurrentItem() }, []) const HIDE_SCROLLBAR = css` diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/LiquidDetailCard.test.tsx b/app/src/organisms/LiquidsLabwareDetailsModal/__tests__/LiquidDetailCard.test.tsx similarity index 98% rename from app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/LiquidDetailCard.test.tsx rename to app/src/organisms/LiquidsLabwareDetailsModal/__tests__/LiquidDetailCard.test.tsx index 8ed860659a5..a96c8128897 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/LiquidDetailCard.test.tsx +++ b/app/src/organisms/LiquidsLabwareDetailsModal/__tests__/LiquidDetailCard.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach, vi, expect } from 'vitest' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/LiquidsLabwareDetailsModal.test.tsx b/app/src/organisms/LiquidsLabwareDetailsModal/__tests__/LiquidsLabwareDetailsModal.test.tsx similarity index 89% rename from app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/LiquidsLabwareDetailsModal.test.tsx rename to app/src/organisms/LiquidsLabwareDetailsModal/__tests__/LiquidsLabwareDetailsModal.test.tsx index 6ff268720a4..54b8239da47 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/LiquidsLabwareDetailsModal.test.tsx +++ b/app/src/organisms/LiquidsLabwareDetailsModal/__tests__/LiquidsLabwareDetailsModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, beforeEach, vi, afterEach, expect } from 'vitest' import { screen } from '@testing-library/react' @@ -8,14 +8,16 @@ import { parseLiquidsInLoadOrder } from '@opentrons/shared-data' import { nestedTextMatcher, renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' import { getIsOnDevice } from '/app/redux/config' -import { useMostRecentCompletedAnalysis } from '../../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' import { mockDefinition } from '/app/redux/custom-labware/__fixtures__' -import { getLocationInfoNames } from '../../utils/getLocationInfoNames' -import { getSlotLabwareDefinition } from '../../utils/getSlotLabwareDefinition' +import { + getLocationInfoNames, + getSlotLabwareDefinition, +} from '/app/transformations/commands' import { getLiquidsByIdForLabware, getDisabledWellFillFromLabwareId, -} from '../utils' +} from '/app/transformations/analysis' import { LiquidsLabwareDetailsModal } from '../LiquidsLabwareDetailsModal' import { LiquidDetailCard } from '../LiquidDetailCard' @@ -37,11 +39,9 @@ vi.mock('@opentrons/shared-data', async importOriginal => { } }) vi.mock('/app/redux/config') -vi.mock('../../../../LabwarePositionCheck/useMostRecentCompletedAnalysis') -vi.mock('../../../../Devices/hooks') -vi.mock('../../utils/getLocationInfoNames') -vi.mock('../../utils/getSlotLabwareDefinition') -vi.mock('../utils') +vi.mock('/app/resources/runs') +vi.mock('/app/transformations/commands') +vi.mock('/app/transformations/analysis') vi.mock('../LiquidDetailCard') const render = ( diff --git a/app/src/organisms/LiquidsLabwareDetailsModal/index.ts b/app/src/organisms/LiquidsLabwareDetailsModal/index.ts new file mode 100644 index 00000000000..c548c0283a2 --- /dev/null +++ b/app/src/organisms/LiquidsLabwareDetailsModal/index.ts @@ -0,0 +1 @@ +export * from './LiquidsLabwareDetailsModal' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/ChooseModuleToConfigureModal.tsx b/app/src/organisms/LocationConflictModal/ChooseModuleToConfigureModal.tsx similarity index 97% rename from app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/ChooseModuleToConfigureModal.tsx rename to app/src/organisms/LocationConflictModal/ChooseModuleToConfigureModal.tsx index 96b165742e9..b91a63c7610 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/ChooseModuleToConfigureModal.tsx +++ b/app/src/organisms/LocationConflictModal/ChooseModuleToConfigureModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import { useNavigate } from 'react-router-dom' @@ -24,9 +23,9 @@ import { MAGNETIC_BLOCK_V1, getModuleDisplayName, } from '@opentrons/shared-data' -import { getTopPortalEl } from '../../../../App/portal' +import { getTopPortalEl } from '/app/App/portal' import { OddModal } from '/app/molecules/OddModal' -import { FixtureOption } from '../../../DeviceDetailsDeckConfiguration/AddFixtureModal' +import { FixtureOption } from '/app/organisms/DeviceDetailsDeckConfiguration/AddFixtureModal' import { useNotifyDeckConfigurationQuery } from '/app/resources/deck_configuration' import { SmallButton } from '/app/atoms/buttons' import { useCloseCurrentRun } from '/app/resources/runs' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/LocationConflictModal.tsx b/app/src/organisms/LocationConflictModal/LocationConflictModal.tsx similarity index 97% rename from app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/LocationConflictModal.tsx rename to app/src/organisms/LocationConflictModal/LocationConflictModal.tsx index 660b29c6a5a..f5c272ea673 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/LocationConflictModal.tsx +++ b/app/src/organisms/LocationConflictModal/LocationConflictModal.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { createPortal } from 'react-dom' import { Trans, useTranslation } from 'react-i18next' import { useUpdateDeckConfigurationMutation } from '@opentrons/react-api-client' @@ -32,7 +32,7 @@ import { THERMOCYCLER_V2_REAR_FIXTURE, } from '@opentrons/shared-data' -import { getTopPortalEl } from '../../../../App/portal' +import { getTopPortalEl } from '/app/App/portal' import { OddModal } from '/app/molecules/OddModal' import { SmallButton } from '/app/atoms/buttons/SmallButton' import { useNotifyDeckConfigurationQuery } from '/app/resources/deck_configuration' @@ -72,7 +72,7 @@ export const LocationConflictModal = ( } = props const { t, i18n } = useTranslation(['protocol_setup', 'shared']) - const [showModuleSelect, setShowModuleSelect] = React.useState(false) + const [showModuleSelect, setShowModuleSelect] = useState(false) const deckConfig = useNotifyDeckConfigurationQuery().data ?? [] const { updateDeckConfiguration } = useUpdateDeckConfigurationMutation() const deckConfigurationAtLocationFixtureId = deckConfig.find( @@ -191,9 +191,10 @@ export const LocationConflictModal = ( protocolSpecifiesDisplayName = getModuleDisplayName(requiredModule) } - const displaySlotName = isThermocyclerRequired - ? 'A1 + B1' - : getCutoutDisplayName(cutoutId) + const displaySlotName = + isThermocyclerRequired || isThermocyclerCurrentFixture + ? 'A1 + B1' + : getCutoutDisplayName(cutoutId) if (showModuleSelect && requiredModule != null) { return createPortal( @@ -232,7 +233,7 @@ export const LocationConflictModal = ( } values={{ currentFixture: currentFixtureDisplayName, - cutout: getCutoutDisplayName(cutoutId), + cutout: displaySlotName, }} components={{ block: , @@ -341,7 +342,7 @@ export const LocationConflictModal = ( } values={{ currentFixture: currentFixtureDisplayName, - cutout: getCutoutDisplayName(cutoutId), + cutout: displaySlotName, }} components={{ block: , diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/LocationConflictModal.test.tsx b/app/src/organisms/LocationConflictModal/__tests__/LocationConflictModal.test.tsx similarity index 99% rename from app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/LocationConflictModal.test.tsx rename to app/src/organisms/LocationConflictModal/__tests__/LocationConflictModal.test.tsx index dfc61ee1bc7..207caa02a1b 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/__tests__/LocationConflictModal.test.tsx +++ b/app/src/organisms/LocationConflictModal/__tests__/LocationConflictModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { screen, fireEvent } from '@testing-library/react' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/LocationConflictModal/index.ts b/app/src/organisms/LocationConflictModal/index.ts new file mode 100644 index 00000000000..732157f3a5a --- /dev/null +++ b/app/src/organisms/LocationConflictModal/index.ts @@ -0,0 +1,2 @@ +export * from './LocationConflictModal' +export * from './ChooseModuleToConfigureModal' diff --git a/app/src/organisms/ModuleCard/AboutModuleSlideout.tsx b/app/src/organisms/ModuleCard/AboutModuleSlideout.tsx index 61cf06a970c..8fedee8299a 100644 --- a/app/src/organisms/ModuleCard/AboutModuleSlideout.tsx +++ b/app/src/organisms/ModuleCard/AboutModuleSlideout.tsx @@ -1,9 +1,10 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' import { RUN_STATUS_RUNNING, RUN_STATUS_FINISHING } from '@opentrons/api-client' import { ALIGN_START, + Banner, Btn, COLORS, DIRECTION_COLUMN, @@ -15,17 +16,16 @@ import { TYPOGRAPHY, } from '@opentrons/components' import { getModuleDisplayName } from '@opentrons/shared-data' -import { Slideout } from '/app/atoms/Slideout' -import { Banner } from '/app/atoms/Banner' +import { Slideout } from '../../atoms/Slideout' import { useCurrentRunStatus } from '../RunTimeControl/hooks' import type { AttachedModule } from '/app/redux/modules/types' interface AboutModuleSlideoutProps { module: AttachedModule - onCloseClick: () => unknown + onCloseClick: () => void isExpanded: boolean - firmwareUpdateClick: () => unknown + firmwareUpdateClick: () => void } const ALERT_ITEM_STYLE = css` @@ -40,7 +40,7 @@ export const AboutModuleSlideout = ( const { i18n, t } = useTranslation(['device_details', 'shared']) const moduleName = getModuleDisplayName(module.moduleModel) const runStatus = useCurrentRunStatus() - const [showBanner, setShowBanner] = React.useState(true) + const [showBanner, setShowBanner] = useState(true) const isDisabled = runStatus === RUN_STATUS_RUNNING || runStatus === RUN_STATUS_FINISHING diff --git a/app/src/organisms/ModuleCard/AbsorbanceReaderData.tsx b/app/src/organisms/ModuleCard/AbsorbanceReaderData.tsx index 7a6a77d0ce3..72deca8bcac 100644 --- a/app/src/organisms/ModuleCard/AbsorbanceReaderData.tsx +++ b/app/src/organisms/ModuleCard/AbsorbanceReaderData.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { TYPOGRAPHY, LegacyStyledText } from '@opentrons/components' import type { AbsorbanceReaderModule } from '/app/redux/modules/types' diff --git a/app/src/organisms/ModuleCard/AbsorbanceReaderSlideout.tsx b/app/src/organisms/ModuleCard/AbsorbanceReaderSlideout.tsx index 5ef3ffd3845..8cfd8597135 100644 --- a/app/src/organisms/ModuleCard/AbsorbanceReaderSlideout.tsx +++ b/app/src/organisms/ModuleCard/AbsorbanceReaderSlideout.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { getModuleDisplayName } from '@opentrons/shared-data' import { SPACING, LegacyStyledText, TYPOGRAPHY } from '@opentrons/components' diff --git a/app/src/organisms/ModuleCard/Collapsible.tsx b/app/src/organisms/ModuleCard/Collapsible.tsx index 44b7c883b8a..cc15a88d4a0 100644 --- a/app/src/organisms/ModuleCard/Collapsible.tsx +++ b/app/src/organisms/ModuleCard/Collapsible.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { ALIGN_CENTER, diff --git a/app/src/organisms/ModuleCard/ErrorInfo.tsx b/app/src/organisms/ModuleCard/ErrorInfo.tsx index 57caacbdc56..77476e23307 100644 --- a/app/src/organisms/ModuleCard/ErrorInfo.tsx +++ b/app/src/organisms/ModuleCard/ErrorInfo.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import { @@ -8,6 +8,7 @@ import { } from '@opentrons/shared-data' import { ALIGN_START, + Banner, Btn, DIRECTION_COLUMN, DIRECTION_ROW, @@ -19,7 +20,6 @@ import { LegacyStyledText, TYPOGRAPHY, } from '@opentrons/components' -import { Banner } from '/app/atoms/Banner' import { getTopPortalEl } from '../../App/portal' import type { AttachedModule } from '/app/redux/modules/types' @@ -30,7 +30,7 @@ interface ErrorInfoProps { export function ErrorInfo(props: ErrorInfoProps): JSX.Element | null { const { attachedModule } = props const { t } = useTranslation(['device_details', 'shared', 'branded']) - const [showErrorDetails, setShowErrorDetails] = React.useState(false) + const [showErrorDetails, setShowErrorDetails] = useState(false) let isError: boolean = false // extend this logic when we know how to tell when Mag/Temp modules are in error state diff --git a/app/src/organisms/ModuleCard/FirmwareUpdateFailedModal.tsx b/app/src/organisms/ModuleCard/FirmwareUpdateFailedModal.tsx index 180e9574b72..47e95f8bf96 100644 --- a/app/src/organisms/ModuleCard/FirmwareUpdateFailedModal.tsx +++ b/app/src/organisms/ModuleCard/FirmwareUpdateFailedModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { getModuleDisplayName } from '@opentrons/shared-data' import { diff --git a/app/src/organisms/ModuleCard/HeaterShakerModuleData.tsx b/app/src/organisms/ModuleCard/HeaterShakerModuleData.tsx index 668f8f66fd0..67bed6f6e5d 100644 --- a/app/src/organisms/ModuleCard/HeaterShakerModuleData.tsx +++ b/app/src/organisms/ModuleCard/HeaterShakerModuleData.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { COLORS, diff --git a/app/src/organisms/ModuleCard/MagneticModuleData.tsx b/app/src/organisms/ModuleCard/MagneticModuleData.tsx index 580731eeab8..f9282505b8a 100644 --- a/app/src/organisms/ModuleCard/MagneticModuleData.tsx +++ b/app/src/organisms/ModuleCard/MagneticModuleData.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { COLORS, TYPOGRAPHY, LegacyStyledText } from '@opentrons/components' import { MAGNETIC_MODULE_V2 } from '@opentrons/shared-data' diff --git a/app/src/organisms/ModuleCard/MagneticModuleSlideout.tsx b/app/src/organisms/ModuleCard/MagneticModuleSlideout.tsx index b4fc4dac9bc..66e428c682d 100644 --- a/app/src/organisms/ModuleCard/MagneticModuleSlideout.tsx +++ b/app/src/organisms/ModuleCard/MagneticModuleSlideout.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { useCreateLiveCommandMutation } from '@opentrons/react-api-client' @@ -73,9 +73,9 @@ export const MagneticModuleSlideout = ( const { module, isExpanded, onCloseClick } = props const { t } = useTranslation('device_details') const { createLiveCommand } = useCreateLiveCommandMutation() - const [engageHeightValue, setEngageHeightValue] = React.useState< - string | null - >(null) + const [engageHeightValue, setEngageHeightValue] = useState( + null + ) const moduleName = getModuleDisplayName(module.moduleModel) const info = getInfoByModel(module.moduleModel) diff --git a/app/src/organisms/ModuleCard/ModuleOverflowMenu.tsx b/app/src/organisms/ModuleCard/ModuleOverflowMenu.tsx index fe27d5376c6..a451846c2fb 100644 --- a/app/src/organisms/ModuleCard/ModuleOverflowMenu.tsx +++ b/app/src/organisms/ModuleCard/ModuleOverflowMenu.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { Fragment } from 'react' import { useTranslation } from 'react-i18next' import { @@ -148,7 +148,7 @@ export const ModuleOverflowMenu = ( {menuOverflowItemsByModuleType[module.moduleType].map( (item: any, index: number) => { return ( - + item.onClick(item.isSecondary)} disabled={item.disabledReason || isDisabled} @@ -157,7 +157,7 @@ export const ModuleOverflowMenu = ( {item.setSetting} {item.menuButtons} - + ) } )} diff --git a/app/src/organisms/ModuleCard/ModuleSetupModal.tsx b/app/src/organisms/ModuleCard/ModuleSetupModal.tsx index 218bfe4dd6b..e4a9d6d458d 100644 --- a/app/src/organisms/ModuleCard/ModuleSetupModal.tsx +++ b/app/src/organisms/ModuleCard/ModuleSetupModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { createPortal } from 'react-dom' import code from '/app/assets/images/module_instruction_code.png' diff --git a/app/src/organisms/ModuleCard/TemperatureModuleData.tsx b/app/src/organisms/ModuleCard/TemperatureModuleData.tsx index 369b7999552..a5ba2bcbd9f 100644 --- a/app/src/organisms/ModuleCard/TemperatureModuleData.tsx +++ b/app/src/organisms/ModuleCard/TemperatureModuleData.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { COLORS, diff --git a/app/src/organisms/ModuleCard/TemperatureModuleSlideout.tsx b/app/src/organisms/ModuleCard/TemperatureModuleSlideout.tsx index c3445efaf06..18928ecae10 100644 --- a/app/src/organisms/ModuleCard/TemperatureModuleSlideout.tsx +++ b/app/src/organisms/ModuleCard/TemperatureModuleSlideout.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { useCreateLiveCommandMutation } from '@opentrons/react-api-client' @@ -37,9 +37,7 @@ export const TemperatureModuleSlideout = ( const { t } = useTranslation('device_details') const { createLiveCommand } = useCreateLiveCommandMutation() const name = getModuleDisplayName(module.moduleModel) - const [temperatureValue, setTemperatureValue] = React.useState( - null - ) + const [temperatureValue, setTemperatureValue] = useState(null) const handleSubmitTemperature = (): void => { if (temperatureValue != null) { const saveTempCommand: TemperatureModuleSetTargetTemperatureCreateCommand = { diff --git a/app/src/organisms/ModuleCard/TestShakeSlideout.tsx b/app/src/organisms/ModuleCard/TestShakeSlideout.tsx index 53847cddea3..d7bfb0e97a9 100644 --- a/app/src/organisms/ModuleCard/TestShakeSlideout.tsx +++ b/app/src/organisms/ModuleCard/TestShakeSlideout.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' @@ -66,11 +66,10 @@ export const TestShakeSlideout = ( }) const { toggleLatch, isLatchClosed } = useLatchControls(module) const configHasHeaterShakerAttached = useSelector(getIsHeaterShakerAttached) - const [shakeValue, setShakeValue] = React.useState(null) - const [ - showModuleSetupModal, - setShowModuleSetupModal, - ] = React.useState(false) + const [shakeValue, setShakeValue] = useState(null) + const [showModuleSetupModal, setShowModuleSetupModal] = useState( + false + ) const isShaking = module.data.speedStatus !== 'idle' const setShakeCommand: HeaterShakerSetAndWaitForShakeSpeedCreateCommand = { diff --git a/app/src/organisms/ModuleCard/ThermocyclerModuleData.tsx b/app/src/organisms/ModuleCard/ThermocyclerModuleData.tsx index f3d7d910301..c0eed06384d 100644 --- a/app/src/organisms/ModuleCard/ThermocyclerModuleData.tsx +++ b/app/src/organisms/ModuleCard/ThermocyclerModuleData.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { StatusLabel } from '/app/atoms/StatusLabel' import { diff --git a/app/src/organisms/ModuleCard/ThermocyclerModuleSlideout.tsx b/app/src/organisms/ModuleCard/ThermocyclerModuleSlideout.tsx index c393336357e..e0bf348022c 100644 --- a/app/src/organisms/ModuleCard/ThermocyclerModuleSlideout.tsx +++ b/app/src/organisms/ModuleCard/ThermocyclerModuleSlideout.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { CELSIUS, @@ -40,7 +40,7 @@ export const ThermocyclerModuleSlideout = ( ): JSX.Element | null => { const { module, onCloseClick, isExpanded, isSecondaryTemp } = props const { t } = useTranslation('device_details') - const [tempValue, setTempValue] = React.useState(null) + const [tempValue, setTempValue] = useState(null) const { createLiveCommand } = useCreateLiveCommandMutation() const moduleName = getModuleDisplayName(module.moduleModel) const modulePart = isSecondaryTemp ? 'Lid' : 'Block' diff --git a/app/src/organisms/ModuleCard/__tests__/AboutModuleSlideout.test.tsx b/app/src/organisms/ModuleCard/__tests__/AboutModuleSlideout.test.tsx index 19f49c412c8..c7f90a42b23 100644 --- a/app/src/organisms/ModuleCard/__tests__/AboutModuleSlideout.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/AboutModuleSlideout.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ModuleCard/__tests__/Collapsible.test.tsx b/app/src/organisms/ModuleCard/__tests__/Collapsible.test.tsx index e312a177459..3db479e3228 100644 --- a/app/src/organisms/ModuleCard/__tests__/Collapsible.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/Collapsible.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ModuleCard/__tests__/ConfirmAttachmentModal.test.tsx b/app/src/organisms/ModuleCard/__tests__/ConfirmAttachmentModal.test.tsx index 73e8f495e9c..ccc81bcb167 100644 --- a/app/src/organisms/ModuleCard/__tests__/ConfirmAttachmentModal.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/ConfirmAttachmentModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/ModuleCard/__tests__/ErrorInfo.test.tsx b/app/src/organisms/ModuleCard/__tests__/ErrorInfo.test.tsx index 467f512ef08..bde80a0d7d2 100644 --- a/app/src/organisms/ModuleCard/__tests__/ErrorInfo.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/ErrorInfo.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { beforeEach, describe, expect, it } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/ModuleCard/__tests__/FirmwareUpdateFailedModal.test.tsx b/app/src/organisms/ModuleCard/__tests__/FirmwareUpdateFailedModal.test.tsx index 498483c7b67..13395bcff69 100644 --- a/app/src/organisms/ModuleCard/__tests__/FirmwareUpdateFailedModal.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/FirmwareUpdateFailedModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { beforeEach, describe, expect, it, vi } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/ModuleCard/__tests__/HeaterShakerModuleData.test.tsx b/app/src/organisms/ModuleCard/__tests__/HeaterShakerModuleData.test.tsx index 7128c46f55d..348fdb614d4 100644 --- a/app/src/organisms/ModuleCard/__tests__/HeaterShakerModuleData.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/HeaterShakerModuleData.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/ModuleCard/__tests__/HeaterShakerSlideout.test.tsx b/app/src/organisms/ModuleCard/__tests__/HeaterShakerSlideout.test.tsx index 74f770bb235..883d5b6bb7c 100644 --- a/app/src/organisms/ModuleCard/__tests__/HeaterShakerSlideout.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/HeaterShakerSlideout.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ModuleCard/__tests__/MagneticModuleData.test.tsx b/app/src/organisms/ModuleCard/__tests__/MagneticModuleData.test.tsx index 594b17f1df0..b6534d233e3 100644 --- a/app/src/organisms/ModuleCard/__tests__/MagneticModuleData.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/MagneticModuleData.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { afterEach, beforeEach, describe, it, vi } from 'vitest' diff --git a/app/src/organisms/ModuleCard/__tests__/MagneticModuleSlideout.test.tsx b/app/src/organisms/ModuleCard/__tests__/MagneticModuleSlideout.test.tsx index 2820cb7651c..fa10546af90 100644 --- a/app/src/organisms/ModuleCard/__tests__/MagneticModuleSlideout.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/MagneticModuleSlideout.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' import { COLORS } from '@opentrons/components' diff --git a/app/src/organisms/ModuleCard/__tests__/ModuleCard.test.tsx b/app/src/organisms/ModuleCard/__tests__/ModuleCard.test.tsx index e426c310216..a0628e1ddc2 100644 --- a/app/src/organisms/ModuleCard/__tests__/ModuleCard.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/ModuleCard.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { fireEvent, screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ModuleCard/__tests__/ModuleOverflowMenu.test.tsx b/app/src/organisms/ModuleCard/__tests__/ModuleOverflowMenu.test.tsx index c840dfe1cff..0bb611f585e 100644 --- a/app/src/organisms/ModuleCard/__tests__/ModuleOverflowMenu.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/ModuleOverflowMenu.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ModuleCard/__tests__/ModuleSetupModal.test.tsx b/app/src/organisms/ModuleCard/__tests__/ModuleSetupModal.test.tsx index 3a8dab77008..7f24a60bc7c 100644 --- a/app/src/organisms/ModuleCard/__tests__/ModuleSetupModal.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/ModuleSetupModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ModuleCard/__tests__/TemperatureModuleData.test.tsx b/app/src/organisms/ModuleCard/__tests__/TemperatureModuleData.test.tsx index 603388807c3..5b851ce5796 100644 --- a/app/src/organisms/ModuleCard/__tests__/TemperatureModuleData.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/TemperatureModuleData.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ModuleCard/__tests__/TemperatureModuleSlideout.test.tsx b/app/src/organisms/ModuleCard/__tests__/TemperatureModuleSlideout.test.tsx index 65fd489abc6..ce65306741d 100644 --- a/app/src/organisms/ModuleCard/__tests__/TemperatureModuleSlideout.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/TemperatureModuleSlideout.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ModuleCard/__tests__/TestShakeSlideout.test.tsx b/app/src/organisms/ModuleCard/__tests__/TestShakeSlideout.test.tsx index 02e3509b078..f11816df8b6 100644 --- a/app/src/organisms/ModuleCard/__tests__/TestShakeSlideout.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/TestShakeSlideout.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen, waitFor } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ModuleCard/__tests__/ThermocyclerModuleData.test.tsx b/app/src/organisms/ModuleCard/__tests__/ThermocyclerModuleData.test.tsx index f4289bf175c..0885c74bb5d 100644 --- a/app/src/organisms/ModuleCard/__tests__/ThermocyclerModuleData.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/ThermocyclerModuleData.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ModuleCard/__tests__/ThermocyclerModuleSlideout.test.tsx b/app/src/organisms/ModuleCard/__tests__/ThermocyclerModuleSlideout.test.tsx index 5996f43a6ab..d93a1b1f607 100644 --- a/app/src/organisms/ModuleCard/__tests__/ThermocyclerModuleSlideout.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/ThermocyclerModuleSlideout.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ModuleCard/__tests__/hooks.test.tsx b/app/src/organisms/ModuleCard/__tests__/hooks.test.tsx index 0a92dd08099..f363eb7c0e0 100644 --- a/app/src/organisms/ModuleCard/__tests__/hooks.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/hooks.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Provider } from 'react-redux' import { when } from 'vitest-when' import { createStore } from 'redux' @@ -18,8 +18,11 @@ import { mockThermocyclerGen2, } from '/app/redux/modules/__fixtures__' import { useIsRobotBusy, useRunStatuses } from '../../Devices/hooks' -import { useMostRecentCompletedAnalysis } from '../../LabwarePositionCheck/useMostRecentCompletedAnalysis' -import { useCurrentRunId } from '/app/resources/runs' + +import { + useCurrentRunId, + useMostRecentCompletedAnalysis, +} from '/app/resources/runs' import { useLatchControls, useModuleOverflowMenu, @@ -30,7 +33,6 @@ import type { Store } from 'redux' import type { State } from '/app/redux/types' vi.mock('@opentrons/react-api-client') -vi.mock('../../LabwarePositionCheck/useMostRecentCompletedAnalysis') vi.mock('/app/resources/runs') vi.mock('../../Devices/hooks') diff --git a/app/src/organisms/ModuleCard/hooks.tsx b/app/src/organisms/ModuleCard/hooks.tsx index 0507fb80d88..b0322e9ebc1 100644 --- a/app/src/organisms/ModuleCard/hooks.tsx +++ b/app/src/organisms/ModuleCard/hooks.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useCreateLiveCommandMutation } from '@opentrons/react-api-client' import { useTranslation } from 'react-i18next' import { @@ -13,8 +12,11 @@ import { TEMPERATURE_MODULE_TYPE, THERMOCYCLER_MODULE_TYPE, } from '@opentrons/shared-data' -import { useMostRecentCompletedAnalysis } from '../LabwarePositionCheck/useMostRecentCompletedAnalysis' -import { useCurrentRunId } from '/app/resources/runs' + +import { + useCurrentRunId, + useMostRecentCompletedAnalysis, +} from '/app/resources/runs' import type { HeaterShakerCloseLatchCreateCommand, diff --git a/app/src/organisms/ModuleCard/index.tsx b/app/src/organisms/ModuleCard/index.tsx index f8f847c2a93..d2d2dd1b838 100644 --- a/app/src/organisms/ModuleCard/index.tsx +++ b/app/src/organisms/ModuleCard/index.tsx @@ -1,9 +1,10 @@ -import * as React from 'react' +import { useState } from 'react' import { Trans, useTranslation } from 'react-i18next' import { useDispatch, useSelector } from 'react-redux' import { ALIGN_START, + Banner, BORDERS, Box, COLORS, @@ -41,12 +42,11 @@ import { dismissRequest, SUCCESS, } from '/app/redux/robot-api' -import { Banner } from '/app/atoms/Banner' import { UpdateBanner } from '/app/molecules/UpdateBanner' import { useChainLiveCommands } from '/app/resources/runs' import { useCurrentRunStatus } from '/app/organisms/RunTimeControl/hooks' import { useIsFlex } from '/app/redux-resources/robots' -import { getModuleTooHot } from '/app/organisms/Devices/getModuleTooHot' +import { getModuleTooHot } from '/app/transformations/modules' import { useToaster } from '/app/organisms/ToasterOven' import { MagneticModuleData } from './MagneticModuleData' import { TemperatureModuleData } from './TemperatureModuleData' @@ -60,7 +60,7 @@ import { HeaterShakerModuleData } from './HeaterShakerModuleData' import { HeaterShakerSlideout } from './HeaterShakerSlideout' import { TestShakeSlideout } from './TestShakeSlideout' import { ModuleWizardFlows } from '../ModuleWizardFlows' -import { getModulePrepCommands } from '/app/organisms/Devices/getModulePrepCommands' +import { getModulePrepCommands } from '/app/local-resources/modules' import { getModuleCardImage } from './utils' import { FirmwareUpdateFailedModal } from './FirmwareUpdateFailedModal' import { ErrorInfo } from './ErrorInfo' @@ -116,13 +116,13 @@ export const ModuleCard = (props: ModuleCardProps): JSX.Element | null => { setShowOverflowMenu(false) }, }) - const [showSlideout, setShowSlideout] = React.useState(false) - const [hasSecondary, setHasSecondary] = React.useState(false) - const [showAboutModule, setShowAboutModule] = React.useState(false) - const [showTestShake, setShowTestShake] = React.useState(false) - const [showHSWizard, setShowHSWizard] = React.useState(false) - const [showFWBanner, setShowFWBanner] = React.useState(true) - const [showCalModal, setShowCalModal] = React.useState(false) + const [showSlideout, setShowSlideout] = useState(false) + const [hasSecondary, setHasSecondary] = useState(false) + const [showAboutModule, setShowAboutModule] = useState(false) + const [showTestShake, setShowTestShake] = useState(false) + const [showHSWizard, setShowHSWizard] = useState(false) + const [showFWBanner, setShowFWBanner] = useState(true) + const [showCalModal, setShowCalModal] = useState(false) const [targetProps, tooltipProps] = useHoverTooltip() @@ -143,7 +143,7 @@ export const ModuleCard = (props: ModuleCardProps): JSX.Element | null => { const hasUpdated = !module.hasAvailableUpdate && latestRequest?.status === SUCCESS - const [showFirmwareToast, setShowFirmwareToast] = React.useState(hasUpdated) + const [showFirmwareToast, setShowFirmwareToast] = useState(hasUpdated) const { makeToast } = useToaster() if (showFirmwareToast) { makeToast(t('firmware_updated_successfully') as string, SUCCESS_TOAST) @@ -240,7 +240,7 @@ export const ModuleCard = (props: ModuleCardProps): JSX.Element | null => { const [ prepCommandErrorMessage, setPrepCommandErrorMessage, - ] = React.useState('') + ] = useState('') const handleCalibrateClick = (): void => { if (getModulePrepCommands(module).length > 0) { chainLiveCommands(getModulePrepCommands(module), false).catch( diff --git a/app/src/organisms/ModuleCard/utils.ts b/app/src/organisms/ModuleCard/utils.ts index 825a4bb6c6d..f581b379ca6 100644 --- a/app/src/organisms/ModuleCard/utils.ts +++ b/app/src/organisms/ModuleCard/utils.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useCallback } from 'react' import last from 'lodash/last' import { useDispatchApiRequest } from '/app/redux/robot-api' @@ -55,7 +55,7 @@ export function useModuleApiRequests(): [ const [ requestIdsBySerial, setRequestIdsBySerial, - ] = React.useState({}) + ] = useState({}) const handleModuleApiRequests = ( robotName: string, @@ -84,7 +84,7 @@ export function useModuleApiRequests(): [ } } - const getLatestRequestId = React.useCallback( + const getLatestRequestId = useCallback( (serialNumber: string): string | null => { if (serialNumber in requestIdsBySerial) { return last(requestIdsBySerial[serialNumber]) ?? null diff --git a/app/src/organisms/ModuleWizardFlows/AttachProbe.tsx b/app/src/organisms/ModuleWizardFlows/AttachProbe.tsx index 18fc3657732..fcd15766a54 100644 --- a/app/src/organisms/ModuleWizardFlows/AttachProbe.tsx +++ b/app/src/organisms/ModuleWizardFlows/AttachProbe.tsx @@ -1,8 +1,8 @@ -import * as React from 'react' import { css } from 'styled-components' import { Trans, useTranslation } from 'react-i18next' import { Flex, + Banner, RESPONSIVENESS, SPACING, LegacyStyledText, @@ -20,7 +20,6 @@ import type { CutoutId, CutoutFixtureId, } from '@opentrons/shared-data' -import { Banner } from '/app/atoms/Banner' import { GenericWizardTile } from '/app/molecules/GenericWizardTile' import type { ModuleCalibrationWizardStepProps } from './types' diff --git a/app/src/organisms/ModuleWizardFlows/BeforeBeginning.tsx b/app/src/organisms/ModuleWizardFlows/BeforeBeginning.tsx index edc65d0ebc0..440d8880af6 100644 --- a/app/src/organisms/ModuleWizardFlows/BeforeBeginning.tsx +++ b/app/src/organisms/ModuleWizardFlows/BeforeBeginning.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/ModuleWizardFlows/DetachProbe.tsx b/app/src/organisms/ModuleWizardFlows/DetachProbe.tsx index fa82a3b507d..a580d1af861 100644 --- a/app/src/organisms/ModuleWizardFlows/DetachProbe.tsx +++ b/app/src/organisms/ModuleWizardFlows/DetachProbe.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import detachProbe1 from '/app/assets/videos/pipette-wizard-flows/Pipette_Detach_Probe_1.webm' import detachProbe8 from '/app/assets/videos/pipette-wizard-flows/Pipette_Detach_Probe_8.webm' diff --git a/app/src/organisms/ModuleWizardFlows/PlaceAdapter.tsx b/app/src/organisms/ModuleWizardFlows/PlaceAdapter.tsx index f10c1d29fdc..f0af5ae9cd4 100644 --- a/app/src/organisms/ModuleWizardFlows/PlaceAdapter.tsx +++ b/app/src/organisms/ModuleWizardFlows/PlaceAdapter.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useEffect } from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' import { v4 as uuidv4 } from 'uuid' @@ -79,7 +79,7 @@ export const PlaceAdapter = (props: PlaceAdapterProps): JSX.Element | null => { createdMaintenanceRunId, } = props const { t } = useTranslation('module_wizard_flows') - React.useEffect(() => { + useEffect(() => { if (createdMaintenanceRunId == null) { createMaintenanceRun({}) } diff --git a/app/src/organisms/ModuleWizardFlows/SelectLocation.tsx b/app/src/organisms/ModuleWizardFlows/SelectLocation.tsx index c7b9e98005f..0f993bbfe99 100644 --- a/app/src/organisms/ModuleWizardFlows/SelectLocation.tsx +++ b/app/src/organisms/ModuleWizardFlows/SelectLocation.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import isEqual from 'lodash/isEqual' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' @@ -17,6 +16,7 @@ import { SINGLE_SLOT_FIXTURES, } from '@opentrons/shared-data' import { + Banner, DeckConfigurator, RESPONSIVENESS, SIZE_1, @@ -24,7 +24,6 @@ import { LegacyStyledText, TYPOGRAPHY, } from '@opentrons/components' -import { Banner } from '/app/atoms/Banner' import { GenericWizardTile } from '/app/molecules/GenericWizardTile' import type { ModuleCalibrationWizardStepProps } from './types' import type { diff --git a/app/src/organisms/ModuleWizardFlows/Success.tsx b/app/src/organisms/ModuleWizardFlows/Success.tsx index 724ce984dbb..52a69bb06c0 100644 --- a/app/src/organisms/ModuleWizardFlows/Success.tsx +++ b/app/src/organisms/ModuleWizardFlows/Success.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' import { getModuleDisplayName } from '@opentrons/shared-data' diff --git a/app/src/organisms/ModuleWizardFlows/types.ts b/app/src/organisms/ModuleWizardFlows/types.ts index 39fc6e6920e..64127dd125b 100644 --- a/app/src/organisms/ModuleWizardFlows/types.ts +++ b/app/src/organisms/ModuleWizardFlows/types.ts @@ -1,7 +1,7 @@ import type { AttachedModule } from '@opentrons/api-client' import type { FLOWS, SECTIONS } from './constants' import type { CreateCommand } from '@opentrons/shared-data' -import type { PipetteInformation } from '../Devices/hooks' +import type { PipetteInformation } from '/app/redux/pipettes' export type ModuleCalibrationWizardStep = | BeforeBeginningStep diff --git a/app/src/organisms/Navigation/RestartRobotConfirmationModal.tsx b/app/src/organisms/Navigation/RestartRobotConfirmationModal.tsx index 2cb4d794438..eb903ff5919 100644 --- a/app/src/organisms/Navigation/RestartRobotConfirmationModal.tsx +++ b/app/src/organisms/Navigation/RestartRobotConfirmationModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { useDispatch } from 'react-redux' diff --git a/app/src/organisms/Navigation/__tests__/Navigation.test.tsx b/app/src/organisms/Navigation/__tests__/Navigation.test.tsx index 7090e1a9551..5a18c86f7dc 100644 --- a/app/src/organisms/Navigation/__tests__/Navigation.test.tsx +++ b/app/src/organisms/Navigation/__tests__/Navigation.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/Navigation/__tests__/NavigationMenu.test.tsx b/app/src/organisms/Navigation/__tests__/NavigationMenu.test.tsx index 4912161b8f2..291c07c6c39 100644 --- a/app/src/organisms/Navigation/__tests__/NavigationMenu.test.tsx +++ b/app/src/organisms/Navigation/__tests__/NavigationMenu.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/Navigation/__tests__/RestartRobotConfirmationModal.test.tsx b/app/src/organisms/Navigation/__tests__/RestartRobotConfirmationModal.test.tsx index 4f18362879e..8922a4225c2 100644 --- a/app/src/organisms/Navigation/__tests__/RestartRobotConfirmationModal.test.tsx +++ b/app/src/organisms/Navigation/__tests__/RestartRobotConfirmationModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ODD/NameRobot/ConfirmRobotName.tsx b/app/src/organisms/ODD/NameRobot/ConfirmRobotName.tsx index 526d2a77f3d..962a2d10de3 100644 --- a/app/src/organisms/ODD/NameRobot/ConfirmRobotName.tsx +++ b/app/src/organisms/ODD/NameRobot/ConfirmRobotName.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useNavigate } from 'react-router-dom' import { useTranslation } from 'react-i18next' diff --git a/app/src/organisms/ODD/NameRobot/__tests__/ConfirmRobotName.test.tsx b/app/src/organisms/ODD/NameRobot/__tests__/ConfirmRobotName.test.tsx index 195a0764f0e..d9e8521260e 100644 --- a/app/src/organisms/ODD/NameRobot/__tests__/ConfirmRobotName.test.tsx +++ b/app/src/organisms/ODD/NameRobot/__tests__/ConfirmRobotName.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ODD/NetworkSettings/AlternativeSecurityTypeModal.tsx b/app/src/organisms/ODD/NetworkSettings/AlternativeSecurityTypeModal.tsx index d63d34f0667..856423f4125 100644 --- a/app/src/organisms/ODD/NetworkSettings/AlternativeSecurityTypeModal.tsx +++ b/app/src/organisms/ODD/NetworkSettings/AlternativeSecurityTypeModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useNavigate } from 'react-router-dom' diff --git a/app/src/organisms/ODD/NetworkSettings/ConnectingNetwork.tsx b/app/src/organisms/ODD/NetworkSettings/ConnectingNetwork.tsx index 1fbb9811b6c..578abd4240d 100644 --- a/app/src/organisms/ODD/NetworkSettings/ConnectingNetwork.tsx +++ b/app/src/organisms/ODD/NetworkSettings/ConnectingNetwork.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/ODD/NetworkSettings/DisplaySearchNetwork.tsx b/app/src/organisms/ODD/NetworkSettings/DisplaySearchNetwork.tsx index 195c9d7799d..0dbe2cde650 100644 --- a/app/src/organisms/ODD/NetworkSettings/DisplaySearchNetwork.tsx +++ b/app/src/organisms/ODD/NetworkSettings/DisplaySearchNetwork.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/ODD/NetworkSettings/DisplayWifiList.tsx b/app/src/organisms/ODD/NetworkSettings/DisplayWifiList.tsx index 26ba8610229..fcfaa2931bf 100644 --- a/app/src/organisms/ODD/NetworkSettings/DisplayWifiList.tsx +++ b/app/src/organisms/ODD/NetworkSettings/DisplayWifiList.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useNavigate } from 'react-router-dom' import { css } from 'styled-components' diff --git a/app/src/organisms/ODD/NetworkSettings/FailedToConnect.tsx b/app/src/organisms/ODD/NetworkSettings/FailedToConnect.tsx index d5a8f4cf331..e3859e4ed29 100644 --- a/app/src/organisms/ODD/NetworkSettings/FailedToConnect.tsx +++ b/app/src/organisms/ODD/NetworkSettings/FailedToConnect.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/ODD/NetworkSettings/SetWifiCred.tsx b/app/src/organisms/ODD/NetworkSettings/SetWifiCred.tsx index 52cc37bd2cf..c7fd6914717 100644 --- a/app/src/organisms/ODD/NetworkSettings/SetWifiCred.tsx +++ b/app/src/organisms/ODD/NetworkSettings/SetWifiCred.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useRef, useState, memo, useEffect } from 'react' import { useTranslation } from 'react-i18next' import { @@ -31,16 +31,16 @@ export function SetWifiCred({ setPassword, }: SetWifiCredProps): JSX.Element { const { t } = useTranslation(['device_settings', 'shared']) - const keyboardRef = React.useRef(null) - const [showPassword, setShowPassword] = React.useState(false) - const inputRef = React.useRef(null) + const keyboardRef = useRef(null) + const [showPassword, setShowPassword] = useState(false) + const inputRef = useRef(null) const isUnboxingFlowOngoing = useIsUnboxingFlowOngoing() - const MemoizedInput = React.memo(InputField) + const MemoizedInput = memo(InputField) const handleBlur = (): void => { if (inputRef.current != null) inputRef.current?.focus() } - React.useEffect(() => { + useEffect(() => { if (inputRef.current != null) { inputRef.current.focus() } diff --git a/app/src/organisms/ODD/NetworkSettings/WifiConnectionDetails.tsx b/app/src/organisms/ODD/NetworkSettings/WifiConnectionDetails.tsx index e6fd446f129..2af3cbc2977 100644 --- a/app/src/organisms/ODD/NetworkSettings/WifiConnectionDetails.tsx +++ b/app/src/organisms/ODD/NetworkSettings/WifiConnectionDetails.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { useTranslation } from 'react-i18next' import { useDispatch, useSelector } from 'react-redux' import { useNavigate } from 'react-router-dom' @@ -53,9 +53,9 @@ export function WifiConnectionDetails({ const [ showNetworkDetailsModal, setShowNetworkDetailsModal, - ] = React.useState(false) + ] = useState(false) - React.useEffect(() => { + useEffect(() => { dispatch(fetchStatus(robotName)) // eslint-disable-next-line react-hooks/exhaustive-deps }, []) diff --git a/app/src/organisms/ODD/NetworkSettings/__tests__/AlternativeSecurityTypeModal.test.tsx b/app/src/organisms/ODD/NetworkSettings/__tests__/AlternativeSecurityTypeModal.test.tsx index 719ede15b7e..a01af9bba66 100644 --- a/app/src/organisms/ODD/NetworkSettings/__tests__/AlternativeSecurityTypeModal.test.tsx +++ b/app/src/organisms/ODD/NetworkSettings/__tests__/AlternativeSecurityTypeModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ODD/NetworkSettings/__tests__/ConnectingNetwork.test.tsx b/app/src/organisms/ODD/NetworkSettings/__tests__/ConnectingNetwork.test.tsx index 293d6ad31cc..e040bee4572 100644 --- a/app/src/organisms/ODD/NetworkSettings/__tests__/ConnectingNetwork.test.tsx +++ b/app/src/organisms/ODD/NetworkSettings/__tests__/ConnectingNetwork.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { screen } from '@testing-library/react' import { beforeEach, describe, expect, it } from 'vitest' diff --git a/app/src/organisms/ODD/NetworkSettings/__tests__/DisplaySearchNetwork.test.tsx b/app/src/organisms/ODD/NetworkSettings/__tests__/DisplaySearchNetwork.test.tsx index 67a04a01977..b26dba4cd68 100644 --- a/app/src/organisms/ODD/NetworkSettings/__tests__/DisplaySearchNetwork.test.tsx +++ b/app/src/organisms/ODD/NetworkSettings/__tests__/DisplaySearchNetwork.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { describe, expect, it } from 'vitest' diff --git a/app/src/organisms/ODD/NetworkSettings/__tests__/DisplayWifiList.test.tsx b/app/src/organisms/ODD/NetworkSettings/__tests__/DisplayWifiList.test.tsx index 756bc146ed6..e1449be3b9d 100644 --- a/app/src/organisms/ODD/NetworkSettings/__tests__/DisplayWifiList.test.tsx +++ b/app/src/organisms/ODD/NetworkSettings/__tests__/DisplayWifiList.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ODD/NetworkSettings/__tests__/FailedToConnect.test.tsx b/app/src/organisms/ODD/NetworkSettings/__tests__/FailedToConnect.test.tsx index 6bcd02a200e..3dbf7bf1f46 100644 --- a/app/src/organisms/ODD/NetworkSettings/__tests__/FailedToConnect.test.tsx +++ b/app/src/organisms/ODD/NetworkSettings/__tests__/FailedToConnect.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ODD/NetworkSettings/__tests__/SelectAuthenticationType.test.tsx b/app/src/organisms/ODD/NetworkSettings/__tests__/SelectAuthenticationType.test.tsx index a5df5d7ba5e..4611d1be5f0 100644 --- a/app/src/organisms/ODD/NetworkSettings/__tests__/SelectAuthenticationType.test.tsx +++ b/app/src/organisms/ODD/NetworkSettings/__tests__/SelectAuthenticationType.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { MemoryRouter } from 'react-router-dom' import { afterEach, beforeEach, describe, it, vi } from 'vitest' diff --git a/app/src/organisms/ODD/NetworkSettings/__tests__/SetWifiCred.test.tsx b/app/src/organisms/ODD/NetworkSettings/__tests__/SetWifiCred.test.tsx index 4a89304f677..d1a25f069c9 100644 --- a/app/src/organisms/ODD/NetworkSettings/__tests__/SetWifiCred.test.tsx +++ b/app/src/organisms/ODD/NetworkSettings/__tests__/SetWifiCred.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ODD/NetworkSettings/__tests__/SetWifiSsid.test.tsx b/app/src/organisms/ODD/NetworkSettings/__tests__/SetWifiSsid.test.tsx index ab3fa750168..11eab279c37 100644 --- a/app/src/organisms/ODD/NetworkSettings/__tests__/SetWifiSsid.test.tsx +++ b/app/src/organisms/ODD/NetworkSettings/__tests__/SetWifiSsid.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ODD/NetworkSettings/__tests__/WifiConnectionDetails.test.tsx b/app/src/organisms/ODD/NetworkSettings/__tests__/WifiConnectionDetails.test.tsx index 0ac96705adc..efcee37e0c6 100644 --- a/app/src/organisms/ODD/NetworkSettings/__tests__/WifiConnectionDetails.test.tsx +++ b/app/src/organisms/ODD/NetworkSettings/__tests__/WifiConnectionDetails.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ODD/ProtocolDetails/ProtocolDetailsSkeleton.tsx b/app/src/organisms/ODD/ProtocolDetails/ProtocolDetailsSkeleton.tsx index e34e2d92b35..a7fbcc9accf 100644 --- a/app/src/organisms/ODD/ProtocolDetails/ProtocolDetailsSkeleton.tsx +++ b/app/src/organisms/ODD/ProtocolDetails/ProtocolDetailsSkeleton.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - import { Flex, BORDERS, DIRECTION_COLUMN, SPACING } from '@opentrons/components' import { Skeleton } from '/app/atoms/Skeleton' diff --git a/app/src/organisms/ODD/ProtocolDetails/__tests__/ProtocolDetailsSkeleton.test.tsx b/app/src/organisms/ODD/ProtocolDetails/__tests__/ProtocolDetailsSkeleton.test.tsx index 04c377834ef..c56d09722fe 100644 --- a/app/src/organisms/ODD/ProtocolDetails/__tests__/ProtocolDetailsSkeleton.test.tsx +++ b/app/src/organisms/ODD/ProtocolDetails/__tests__/ProtocolDetailsSkeleton.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { render, screen } from '@testing-library/react' import { describe, expect, it } from 'vitest' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupDeckConfiguration/__tests__/ProtocolSetupDeckConfiguration.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupDeckConfiguration/__tests__/ProtocolSetupDeckConfiguration.test.tsx index f56c6db8c72..84a7fd2eb87 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupDeckConfiguration/__tests__/ProtocolSetupDeckConfiguration.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupDeckConfiguration/__tests__/ProtocolSetupDeckConfiguration.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, beforeEach, afterEach } from 'vitest' @@ -11,9 +11,9 @@ import { import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' -import { useMostRecentCompletedAnalysis } from '../../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' -import { ProtocolSetupDeckConfiguration } from '..' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' import { useNotifyDeckConfigurationQuery } from '/app/resources/deck_configuration' +import { ProtocolSetupDeckConfiguration } from '..' import type { UseQueryResult } from 'react-query' import type { @@ -24,7 +24,7 @@ import type { Modules } from '@opentrons/api-client' vi.mock('@opentrons/components/src/hardware-sim/BaseDeck/index') vi.mock('@opentrons/react-api-client') -vi.mock('../../../../LabwarePositionCheck/useMostRecentCompletedAnalysis') +vi.mock('/app/resources/runs') vi.mock('/app/resources/deck_configuration') const mockSetSetupScreen = vi.fn() diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupDeckConfiguration/index.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupDeckConfiguration/index.tsx index cc3c181bd2c..f52d620e03a 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupDeckConfiguration/index.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupDeckConfiguration/index.tsx @@ -22,7 +22,7 @@ import { import { ChildNavigation } from '../../../ChildNavigation' import { AddFixtureModal } from '../../../DeviceDetailsDeckConfiguration/AddFixtureModal' import { DeckConfigurationDiscardChangesModal } from '../../../DeviceDetailsDeckConfiguration/DeckConfigurationDiscardChangesModal' -import { useMostRecentCompletedAnalysis } from '../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' import { getTopPortalEl } from '../../../../App/portal' import { useNotifyDeckConfigurationQuery } from '/app/resources/deck_configuration' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupInstruments/ProtocolSetupInstruments.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupInstruments/ProtocolSetupInstruments.tsx index 6c757f16dce..a6377b248bc 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupInstruments/ProtocolSetupInstruments.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupInstruments/ProtocolSetupInstruments.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import styled from 'styled-components' import { useTranslation } from 'react-i18next' import { @@ -13,8 +13,8 @@ import { import { useInstrumentsQuery } from '@opentrons/react-api-client' import { ODDBackButton } from '/app/molecules/ODDBackButton' import { PipetteRecalibrationODDWarning } from '/app/pages/ODD/InstrumentsDashboard/PipetteRecalibrationODDWarning' -import { getShowPipetteCalibrationWarning } from '../../../Devices/utils' -import { useMostRecentCompletedAnalysis } from '../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { getShowPipetteCalibrationWarning } from '/app/transformations/instruments' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' import { ProtocolInstrumentMountItem } from '../../../InstrumentMountItem' import type { GripperData, PipetteData } from '@opentrons/api-client' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupInstruments/__tests__/ProtocolSetupInstruments.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupInstruments/__tests__/ProtocolSetupInstruments.test.tsx index 3d82d3e544d..3a1273130ee 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupInstruments/__tests__/ProtocolSetupInstruments.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupInstruments/__tests__/ProtocolSetupInstruments.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { when } from 'vitest-when' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' @@ -11,13 +10,13 @@ import { import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' -import { useMostRecentCompletedAnalysis } from '/app/organisms/LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' import { useIsOEMMode } from '/app/resources/robot-settings/hooks' import { mockRecentAnalysis } from '../__fixtures__' import { ProtocolSetupInstruments } from '..' vi.mock('@opentrons/react-api-client') -vi.mock('/app/organisms/LabwarePositionCheck/useMostRecentCompletedAnalysis') +vi.mock('/app/resources/runs') vi.mock('/app/resources/robot-settings/hooks') const mockGripperData = { diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/LabwareMapView.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/LabwareMapView.tsx index d7003d8a4fa..b22844cca13 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/LabwareMapView.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/LabwareMapView.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import map from 'lodash/map' import { BaseDeck, Flex } from '@opentrons/components' import { @@ -7,8 +6,8 @@ import { THERMOCYCLER_MODULE_V1, } from '@opentrons/shared-data' -import { getStandardDeckViewLayerBlockList } from '../../../Devices/ProtocolRun/utils/getStandardDeckViewLayerBlockList' -import { getLabwareRenderInfo } from '../../../Devices/ProtocolRun/utils/getLabwareRenderInfo' +import { getStandardDeckViewLayerBlockList } from '/app/local-resources/deck_configuration' +import { getLabwareRenderInfo } from '/app/transformations/analysis' import type { CompletedProtocolAnalysis, diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/SingleLabwareModal.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/SingleLabwareModal.tsx index 81ced71c592..36145dad085 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/SingleLabwareModal.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/SingleLabwareModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import styled from 'styled-components' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/__tests__/LabwareMapView.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/__tests__/LabwareMapView.test.tsx index 541c719def6..8729ae0f811 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/__tests__/LabwareMapView.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/__tests__/LabwareMapView.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { when } from 'vitest-when' import { describe, it, vi, beforeEach, afterEach, expect } from 'vitest' @@ -12,8 +12,8 @@ import { import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' -import { getLabwareRenderInfo } from '../../../../Devices/ProtocolRun/utils/getLabwareRenderInfo' -import { getStandardDeckViewLayerBlockList } from '../../../../Devices/ProtocolRun/utils/getStandardDeckViewLayerBlockList' +import { getLabwareRenderInfo } from '/app/transformations/analysis' +import { getStandardDeckViewLayerBlockList } from '/app/local-resources/deck_configuration' import { mockProtocolModuleInfo } from '../__fixtures__' import { LabwareMapView } from '../LabwareMapView' @@ -25,7 +25,7 @@ import type { ModuleModel, } from '@opentrons/shared-data' -vi.mock('../../../../Devices/ProtocolRun/utils/getLabwareRenderInfo') +vi.mock('/app/transformations/analysis') vi.mock('@opentrons/components/src/hardware-sim/Labware/LabwareRender') vi.mock('@opentrons/components/src/hardware-sim/BaseDeck') vi.mock('@opentrons/shared-data/js/helpers/getSimplestFlexDeckConfig') diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/__tests__/ProtocolSetupLabware.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/__tests__/ProtocolSetupLabware.test.tsx index fe975d4ad0e..08ec91b7e10 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/__tests__/ProtocolSetupLabware.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/__tests__/ProtocolSetupLabware.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { when } from 'vitest-when' import { MemoryRouter } from 'react-router-dom' @@ -15,8 +14,8 @@ import { import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' -import { useMostRecentCompletedAnalysis } from '/app/organisms/LabwarePositionCheck/useMostRecentCompletedAnalysis' -import { getProtocolModulesInfo } from '../../../../Devices/ProtocolRun/utils/getProtocolModulesInfo' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' +import { getProtocolModulesInfo } from '/app/transformations/analysis' import { ProtocolSetupLabware } from '..' import { mockProtocolModuleInfo, @@ -40,8 +39,8 @@ vi.mock('@opentrons/react-api-client', async importOriginal => { } }) -vi.mock('/app/organisms/LabwarePositionCheck/useMostRecentCompletedAnalysis') -vi.mock('../../../../Devices/ProtocolRun/utils/getProtocolModulesInfo') +vi.mock('/app/resources/runs') +vi.mock('/app/transformations/analysis') vi.mock('/app/resources/deck_configuration') const RUN_ID = "otie's run" diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/index.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/index.tsx index 8a08f5fa672..d7563c0dc6d 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/index.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLabware/index.tsx @@ -37,12 +37,14 @@ import { import { FloatingActionButton, SmallButton } from '/app/atoms/buttons' import { ODDBackButton } from '/app/molecules/ODDBackButton' -import { getLabwareSetupItemGroups } from '/app/transformations/commands' +import { + getLabwareSetupItemGroups, + getNestedLabwareInfo, +} from '/app/transformations/commands' +import { getProtocolModulesInfo } from '/app/transformations/analysis' import { useNotifyDeckConfigurationQuery } from '/app/resources/deck_configuration' -import { getProtocolModulesInfo } from '../../../Devices/ProtocolRun/utils/getProtocolModulesInfo' -import { getNestedLabwareInfo } from '../../../Devices/ProtocolRun/SetupLabware/getNestedLabwareInfo' -import { LabwareStackModal } from '../../../Devices/ProtocolRun/SetupLabware/LabwareStackModal' -import { useMostRecentCompletedAnalysis } from '../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { LabwareStackModal } from '/app/molecules/LabwareStackModal' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' import { getAttachedProtocolModuleMatches } from '../ProtocolSetupModulesAndDeck/utils' import { LabwareMapView } from './LabwareMapView' import { SingleLabwareModal } from './SingleLabwareModal' @@ -57,9 +59,11 @@ import type { RunTimeCommand, } from '@opentrons/shared-data' import type { HeaterShakerModule, Modules } from '@opentrons/api-client' -import type { LabwareSetupItem } from '/app/transformations/commands' +import type { + LabwareSetupItem, + NestedLabwareInfo, +} from '/app/transformations/commands' import type { SetupScreens } from '../types' -import type { NestedLabwareInfo } from '../../../Devices/ProtocolRun/SetupLabware/getNestedLabwareInfo' import type { AttachedProtocolModuleMatch } from '../ProtocolSetupModulesAndDeck/utils' const MODULE_REFETCH_INTERVAL_MS = 5000 diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLiquids/LiquidDetails.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLiquids/LiquidDetails.tsx index 0b1651ad0ae..796e9595d85 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLiquids/LiquidDetails.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLiquids/LiquidDetails.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import styled from 'styled-components' import { useTranslation } from 'react-i18next' import { @@ -14,9 +14,9 @@ import { WRAP, } from '@opentrons/components' import { MICRO_LITERS } from '@opentrons/shared-data' -import { LiquidsLabwareDetailsModal } from '../../../Devices/ProtocolRun/SetupLiquids/LiquidsLabwareDetailsModal' -import { getLocationInfoNames } from '../../../Devices/ProtocolRun/utils/getLocationInfoNames' -import { getVolumePerWell } from '../../../Devices/ProtocolRun/SetupLiquids/utils' +import { LiquidsLabwareDetailsModal } from '/app/organisms/LiquidsLabwareDetailsModal' +import { getLocationInfoNames } from '/app/transformations/commands' +import { getVolumePerWell } from '/app/transformations/analysis' import type { LabwareByLiquidId, @@ -70,9 +70,7 @@ interface LiquidDetailsProps { export function LiquidDetails(props: LiquidDetailsProps): JSX.Element { const { liquid, labwareByLiquidId, runId, commands } = props const { t } = useTranslation('protocol_setup') - const [labwareIdModal, setLabwareIdModal] = React.useState( - null - ) + const [labwareIdModal, setLabwareIdModal] = useState(null) return ( diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLiquids/__tests__/LiquidDetails.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLiquids/__tests__/LiquidDetails.test.tsx index 0100a7a27da..feeb3e863a4 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLiquids/__tests__/LiquidDetails.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLiquids/__tests__/LiquidDetails.test.tsx @@ -1,25 +1,23 @@ -import * as React from 'react' +import type * as React from 'react' import { screen, fireEvent } from '@testing-library/react' import { describe, it, beforeEach, vi } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' import { RUN_ID_1 } from '/app/resources/runs/__fixtures__' -import { getLocationInfoNames } from '../../../../Devices/ProtocolRun/utils/getLocationInfoNames' -import { getVolumePerWell } from '../../../../Devices/ProtocolRun/SetupLiquids/utils' +import { getLocationInfoNames } from '/app/transformations/commands' +import { getVolumePerWell } from '/app/transformations/analysis' import { LiquidDetails } from '../LiquidDetails' -import { LiquidsLabwareDetailsModal } from '../../../../Devices/ProtocolRun/SetupLiquids/LiquidsLabwareDetailsModal' +import { LiquidsLabwareDetailsModal } from '/app/organisms/LiquidsLabwareDetailsModal' import { MOCK_LABWARE_INFO_BY_LIQUID_ID, MOCK_PROTOCOL_ANALYSIS, } from '../fixtures' import type { CompletedProtocolAnalysis } from '@opentrons/shared-data' -vi.mock('../../../../Devices/ProtocolRun/SetupLiquids/utils') -vi.mock('../../../../Devices/ProtocolRun/utils/getLocationInfoNames') -vi.mock( - '../../../../Devices/ProtocolRun/SetupLiquids/LiquidsLabwareDetailsModal' -) +vi.mock('/app/transformations/analysis') +vi.mock('/app/transformations/commands') +vi.mock('/app/organisms/LiquidsLabwareDetailsModal') const render = (props: React.ComponentProps) => { return renderWithProviders(, { diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLiquids/__tests__/ProtocolSetupLiquids.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLiquids/__tests__/ProtocolSetupLiquids.test.tsx index 15a705119f5..487fbbd0bce 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLiquids/__tests__/ProtocolSetupLiquids.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLiquids/__tests__/ProtocolSetupLiquids.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, beforeEach, vi } from 'vitest' import { screen, fireEvent } from '@testing-library/react' @@ -10,8 +10,8 @@ import { import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' import { RUN_ID_1 } from '/app/resources/runs/__fixtures__' -import { getTotalVolumePerLiquidId } from '../../../../Devices/ProtocolRun/SetupLiquids/utils' -import { useMostRecentCompletedAnalysis } from '../../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { getTotalVolumePerLiquidId } from '/app/transformations/analysis' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' import { LiquidDetails } from '../LiquidDetails' import { MOCK_LABWARE_INFO_BY_LIQUID_ID, @@ -22,10 +22,10 @@ import { ProtocolSetupLiquids } from '..' import type * as SharedData from '@opentrons/shared-data' -vi.mock('../../../../Devices/ProtocolRun/SetupLiquids/utils') +vi.mock('/app/transformations/analysis') vi.mock('/app/atoms/buttons') vi.mock('../LiquidDetails') -vi.mock('../../../../LabwarePositionCheck/useMostRecentCompletedAnalysis') +vi.mock('/app/resources/runs') vi.mock('@opentrons/shared-data', async importOriginal => { const actualSharedData = await importOriginal() return { diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLiquids/index.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLiquids/index.tsx index f271d312716..551b7e716cd 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLiquids/index.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupLiquids/index.tsx @@ -22,8 +22,8 @@ import { import { ODDBackButton } from '/app/molecules/ODDBackButton' import { SmallButton } from '/app/atoms/buttons' -import { useMostRecentCompletedAnalysis } from '../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' -import { getTotalVolumePerLiquidId } from '../../../Devices/ProtocolRun/SetupLiquids/utils' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' +import { getTotalVolumePerLiquidId } from '/app/transformations/analysis' import { LiquidDetails } from './LiquidDetails' import type { ParsedLiquid, RunTimeCommand } from '@opentrons/shared-data' import type { SetupScreens } from '../types' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/FixtureTable.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/FixtureTable.tsx index 43e5ae29036..3491d7530c7 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/FixtureTable.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/FixtureTable.tsx @@ -20,12 +20,15 @@ import { getFixtureDisplayName, getSimplestDeckConfigForProtocol, SINGLE_SLOT_FIXTURES, + TC_MODULE_LOCATION_OT3, + THERMOCYCLER_V2_FRONT_FIXTURE, + THERMOCYCLER_V2_REAR_FIXTURE, } from '@opentrons/shared-data' import { SmallButton } from '/app/atoms/buttons' import { useDeckConfigurationCompatibility } from '/app/resources/deck_configuration/hooks' import { getRequiredDeckConfig } from '/app/resources/deck_configuration/utils' -import { LocationConflictModal } from '../../../Devices/ProtocolRun/SetupModuleAndDeck/LocationConflictModal' +import { LocationConflictModal } from '/app/organisms/LocationConflictModal' import type { CompletedProtocolAnalysis, @@ -74,8 +77,30 @@ export function FixtureTable({ deckConfigCompatibility ) + const hasTwoLabwareThermocyclerConflicts = + requiredDeckConfigCompatibility.some( + ({ cutoutFixtureId, missingLabwareDisplayName }) => + cutoutFixtureId === THERMOCYCLER_V2_FRONT_FIXTURE && + missingLabwareDisplayName != null + ) && + requiredDeckConfigCompatibility.some( + ({ cutoutFixtureId, missingLabwareDisplayName }) => + cutoutFixtureId === THERMOCYCLER_V2_REAR_FIXTURE && + missingLabwareDisplayName != null + ) + + // if there are two labware conflicts with the thermocycler, don't show the conflict with the thermocycler rear fixture + const filteredDeckConfigCompatibility = requiredDeckConfigCompatibility.filter( + ({ cutoutFixtureId }) => { + return ( + !hasTwoLabwareThermocyclerConflicts || + !(cutoutFixtureId === THERMOCYCLER_V2_REAR_FIXTURE) + ) + } + ) + // list not configured/conflicted fixtures first - const sortedDeckConfigCompatibility = requiredDeckConfigCompatibility.sort( + const sortedDeckConfigCompatibility = filteredDeckConfigCompatibility.sort( a => a.cutoutFixtureId != null && a.compatibleCutoutFixtureIds.includes(a.cutoutFixtureId) @@ -139,6 +164,11 @@ function FixtureTableItem({ cutoutFixtureId != null && compatibleCutoutFixtureIds.includes(cutoutFixtureId) const isRequiredSingleSlotMissing = missingLabwareDisplayName != null + + const isThermocyclerCurrentFixture = + cutoutFixtureId === THERMOCYCLER_V2_FRONT_FIXTURE || + cutoutFixtureId === THERMOCYCLER_V2_REAR_FIXTURE + let chipLabel: JSX.Element if (!isCurrentFixtureCompatible) { const isConflictingFixtureConfigured = @@ -219,7 +249,13 @@ function FixtureTableItem({ - + {showMapView ? ( - + + + ) : ( <> {isModuleMismatch && !clearModuleMismatchBanner ? ( diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/SetupInstructionsModal.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/SetupInstructionsModal.tsx index 4c701bc108e..d2fdc3bf540 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/SetupInstructionsModal.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/SetupInstructionsModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/__tests__/FixtureTable.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/__tests__/FixtureTable.test.tsx index 2f754de1590..e6ca8735d77 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/__tests__/FixtureTable.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/__tests__/FixtureTable.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, beforeEach, afterEach, vi, expect } from 'vitest' @@ -12,7 +12,7 @@ import { import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' -import { LocationConflictModal } from '/app/organisms/Devices/ProtocolRun/SetupModuleAndDeck/LocationConflictModal' +import { LocationConflictModal } from '/app/organisms/LocationConflictModal' import { useDeckConfigurationCompatibility } from '/app/resources/deck_configuration/hooks' import { FixtureTable } from '../FixtureTable' import { getLocalRobot } from '/app/redux/discovery' @@ -20,9 +20,7 @@ import { mockConnectedRobot } from '/app/redux/discovery/__fixtures__' vi.mock('/app/redux/discovery') vi.mock('/app/resources/deck_configuration/hooks') -vi.mock( - '/app/organisms/Devices/ProtocolRun/SetupModuleAndDeck/LocationConflictModal' -) +vi.mock('/app/organisms/LocationConflictModal') const mockSetSetupScreen = vi.fn() const mockSetCutoutId = vi.fn() diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/__tests__/ModulesAndDeckMapView.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/__tests__/ModulesAndDeckMapView.test.tsx index 0fb9b90fd97..d31a0312d02 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/__tests__/ModulesAndDeckMapView.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/__tests__/ModulesAndDeckMapView.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, beforeEach, afterEach } from 'vitest' import { screen } from '@testing-library/react' @@ -16,10 +16,9 @@ vi.mock('@opentrons/components/src/hardware-sim/BaseDeck') vi.mock('@opentrons/api-client') vi.mock('@opentrons/shared-data/js/helpers/getSimplestFlexDeckConfig') vi.mock('/app/redux/config') -vi.mock('../../../../Devices/hooks') vi.mock('/app/resources/deck_configuration/utils') -vi.mock('../../../../Devices/ModuleInfo') -vi.mock('../../../../Devices/ProtocolRun/utils/getLabwareRenderInfo') +vi.mock('/app/molecules/ModuleInfo') +vi.mock('/app/transformations/analysis') const mockRunId = 'mockRunId' const PROTOCOL_ANALYSIS = { diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/__tests__/ProtocolSetupModulesAndDeck.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/__tests__/ProtocolSetupModulesAndDeck.test.tsx index f59e0df775b..4cf2fb65a53 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/__tests__/ProtocolSetupModulesAndDeck.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/__tests__/ProtocolSetupModulesAndDeck.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, screen, waitFor } from '@testing-library/react' import { vi, it, expect, describe, beforeEach, afterEach } from 'vitest' import { when } from 'vitest-when' @@ -13,14 +12,15 @@ import { import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' -import { useChainLiveCommands, useRunStatus } from '/app/resources/runs' -import { mockRobotSideAnalysis } from '/app/molecules/Command/__fixtures__' import { - useAttachedModules, + useChainLiveCommands, + useRunStatus, + useMostRecentCompletedAnalysis, useRunCalibrationStatus, -} from '../../../../Devices/hooks' -import { useMostRecentCompletedAnalysis } from '../../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' -import { getProtocolModulesInfo } from '../../../../Devices/ProtocolRun/utils/getProtocolModulesInfo' +} from '/app/resources/runs' +import { mockRobotSideAnalysis } from '/app/molecules/Command/__fixtures__' +import { useAttachedModules } from '/app/resources/modules' +import { getProtocolModulesInfo } from '/app/transformations/analysis' import { mockApiHeaterShaker } from '/app/redux/modules/__fixtures__' import { mockProtocolModuleInfo } from '../../ProtocolSetupInstruments/__fixtures__' import { getLocalRobot } from '/app/redux/discovery' @@ -29,8 +29,8 @@ import { getAttachedProtocolModuleMatches, getUnmatchedModulesForProtocol, } from '../utils' -import { LocationConflictModal } from '../../../../Devices/ProtocolRun/SetupModuleAndDeck/LocationConflictModal' -import { ModuleWizardFlows } from '../../../../ModuleWizardFlows' +import { LocationConflictModal } from '/app/organisms/LocationConflictModal' +import { ModuleWizardFlows } from '/app/organisms/ModuleWizardFlows' import { SetupInstructionsModal } from '../SetupInstructionsModal' import { FixtureTable } from '../FixtureTable' import { ModulesAndDeckMapView } from '../ModulesAndDeckMapView' @@ -41,18 +41,15 @@ import type { CutoutConfig, DeckConfiguration } from '@opentrons/shared-data' import type { UseQueryResult } from 'react-query' vi.mock('/app/resources/runs') +vi.mock('/app/resources/modules') vi.mock('/app/redux/discovery') -vi.mock('/app/organisms/Devices/hooks') vi.mock('/app/resources/deck_configuration') -vi.mock('/app/organisms/LabwarePositionCheck/useMostRecentCompletedAnalysis') -vi.mock('/app/organisms/Devices/ProtocolRun/utils/getProtocolModulesInfo') +vi.mock('/app/transformations/analysis') vi.mock('../utils') vi.mock('../SetupInstructionsModal') -vi.mock('../../../../ModuleWizardFlows') +vi.mock('/app/organisms/ModuleWizardFlows') vi.mock('../FixtureTable') -vi.mock( - '../../../../Devices/ProtocolRun/SetupModuleAndDeck/LocationConflictModal' -) +vi.mock('/app/organisms/LocationConflictModal') vi.mock('../ModulesAndDeckMapView') const ROBOT_NAME = 'otie' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/__tests__/SetupInstructionsModal.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/__tests__/SetupInstructionsModal.test.tsx index 28c0dd8222b..8f6f4c01739 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/__tests__/SetupInstructionsModal.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/__tests__/SetupInstructionsModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, beforeEach, vi } from 'vitest' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/utils.ts b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/utils.ts index 6d98c8edd11..e272aa5994d 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/utils.ts +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupModulesAndDeck/utils.ts @@ -10,7 +10,7 @@ import { } from '@opentrons/shared-data' import type { DeckConfiguration, RobotType } from '@opentrons/shared-data' -import type { ProtocolModuleInfo } from '../../../Devices/ProtocolRun/utils/getProtocolModulesInfo' +import type { ProtocolModuleInfo } from '/app/transformations/analysis' import type { AttachedModule } from '/app/redux/modules/types' export type AttachedProtocolModuleMatch = ProtocolModuleInfo & { diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupOffsets/index.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupOffsets/index.tsx index a66e6dd8a93..3f287ea80e6 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupOffsets/index.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupOffsets/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { Chip, @@ -16,11 +16,13 @@ import { useToaster } from '/app/organisms/ToasterOven' import { ODDBackButton } from '/app/molecules/ODDBackButton' import { FloatingActionButton, SmallButton } from '/app/atoms/buttons' import type { SetupScreens } from '../types' -import { useMostRecentCompletedAnalysis } from '../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' import { TerseOffsetTable } from '/app/organisms/LabwarePositionCheck/ResultsSummary' import { getLabwareDefinitionsFromCommands } from '/app/molecules/Command/utils/getLabwareDefinitionsFromCommands' -import { useNotifyRunQuery } from '/app/resources/runs' -import { getLatestCurrentOffsets } from '/app/organisms/Devices/ProtocolRun/SetupLabwarePositionCheck/utils' +import { + useNotifyRunQuery, + useMostRecentCompletedAnalysis, +} from '/app/resources/runs' +import { getLatestCurrentOffsets } from '/app/transformations/runs' export interface ProtocolSetupOffsetsProps { runId: string @@ -120,7 +122,7 @@ export function ProtocolSetupOffsets({ /> ) : ( - + )} (initialFileObject) + ] = useState(initialFileObject) const handleBackButton = (): void => { if (!isEqual(csvFileSelected, initialFileObject)) { @@ -87,7 +87,7 @@ export function ChooseCsvFile({ handleGoBack() } - React.useEffect(() => { + useEffect(() => { if (csvFilesOnUSB.length === 0) { setCsvFileSelected({}) } @@ -119,7 +119,7 @@ export function ChooseCsvFile({ {csvFilesOnRobot.length !== 0 ? ( csvFilesOnRobot.map((csv: CsvFileData) => ( - + - + )) ) : ( @@ -150,7 +150,7 @@ export function ChooseCsvFile({ sortedCsvFilesOnUSB.map(csvFilePath => { const fileName = last(csvFilePath.split('/')) return ( - + {csvFilePath.length !== 0 && fileName !== undefined ? ( ) : null} - + ) }) ) : ( diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ChooseEnum.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ChooseEnum.tsx index a4723edf734..5b6ec5e5c5d 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ChooseEnum.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ChooseEnum.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { ALIGN_CENTER, diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ChooseNumber.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ChooseNumber.tsx index 179fbffb5df..934a1591844 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ChooseNumber.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ChooseNumber.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useRef, useState, useEffect } from 'react' import { useTranslation } from 'react-i18next' import { @@ -30,16 +30,14 @@ export function ChooseNumber({ const { makeSnackbar } = useToaster() const { i18n, t } = useTranslation(['protocol_setup', 'shared']) - const keyboardRef = React.useRef(null) - const [paramValue, setParamValue] = React.useState( - String(parameter.value) - ) + const keyboardRef = useRef(null) + const [paramValue, setParamValue] = useState(String(parameter.value)) // We need to arbitrarily set the value of the keyboard to a string the // same length as the initial parameter value (as string) when the component mounts // so that the delete button operates properly on the exisiting input field value. - const [prevKeyboardValue, setPrevKeyboardValue] = React.useState('') - React.useEffect(() => { + const [prevKeyboardValue, setPrevKeyboardValue] = useState('') + useEffect(() => { const arbitraryInput = new Array(paramValue).join('*') // @ts-expect-error keyboard should expose for `setInput` method keyboardRef.current?.setInput(arbitraryInput) diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/EmptyFile.stories.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/EmptyFile.stories.tsx index 45d0fa652fa..5932aca9031 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/EmptyFile.stories.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/EmptyFile.stories.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Flex, SPACING, VIEWPORT } from '@opentrons/components' import { EmptyFile as EmptyFileComponent } from './EmptyFile' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/EmptyFile.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/EmptyFile.tsx index 888a0f0f91b..413042caad8 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/EmptyFile.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/EmptyFile.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { ALIGN_CENTER, diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ProtocolSetupParameters.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ProtocolSetupParameters.tsx index 3264213f10d..bc7c3482d7d 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ProtocolSetupParameters.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ProtocolSetupParameters.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, Fragment } from 'react' import { useTranslation } from 'react-i18next' import { useNavigate } from 'react-router-dom' import { @@ -62,23 +62,20 @@ export function ProtocolSetupParameters({ const [ chooseValueScreen, setChooseValueScreen, - ] = React.useState(null) + ] = useState(null) const [ showNumericalInputScreen, setShowNumericalInputScreen, - ] = React.useState(null) + ] = useState(null) const [ chooseCsvFileScreen, setChooseCsvFileScreen, - ] = React.useState(null) - const [resetValuesModal, showResetValuesModal] = React.useState( - false - ) - const [startSetup, setStartSetup] = React.useState(false) - const [ - runTimeParametersOverrides, - setRunTimeParametersOverrides, - ] = React.useState( + ] = useState(null) + const [resetValuesModal, showResetValuesModal] = useState(false) + const [startSetup, setStartSetup] = useState(false) + const [runTimeParametersOverrides, setRunTimeParametersOverrides] = useState< + RunTimeParameter[] + >( runTimeParameters.map(parameter => parameter.type === 'csv_file' ? { ...parameter, file: null } @@ -296,7 +293,7 @@ export function ProtocolSetupParameters({ setupStatus = 'inform' } return ( - + - + ) } )} diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ResetValuesModal.stories.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ResetValuesModal.stories.tsx index 975d8104a26..3becd83d063 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ResetValuesModal.stories.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ResetValuesModal.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { VIEWPORT } from '@opentrons/components' import { ResetValuesModal } from './ResetValuesModal' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ResetValuesModal.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ResetValuesModal.tsx index 4000052e933..ed3e70cd7df 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ResetValuesModal.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ResetValuesModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ViewOnlyParameters.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ViewOnlyParameters.tsx index c8c892902a4..14956210cfb 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ViewOnlyParameters.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/ViewOnlyParameters.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' @@ -18,7 +18,7 @@ import { LegacyStyledText, TYPOGRAPHY, } from '@opentrons/components' -import { useMostRecentCompletedAnalysis } from '/app/organisms/LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' import { ChildNavigation } from '/app/organisms/ChildNavigation' import { useToaster } from '/app/organisms/ToasterOven' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/AnalysisFailedModal.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/AnalysisFailedModal.test.tsx index b64e1c34e54..ac43f26d621 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/AnalysisFailedModal.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/AnalysisFailedModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, beforeEach, expect } from 'vitest' import { when } from 'vitest-when' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ChooseCsvFile.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ChooseCsvFile.test.tsx index 6440581840c..2f365fa5fbc 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ChooseCsvFile.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ChooseCsvFile.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, beforeEach, vi, expect } from 'vitest' import { screen, fireEvent } from '@testing-library/react' import { when } from 'vitest-when' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ChooseEnum.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ChooseEnum.test.tsx index 9e19d40a78b..a65c760d544 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ChooseEnum.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ChooseEnum.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { it, describe, beforeEach, vi, expect } from 'vitest' import { fireEvent, screen } from '@testing-library/react' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ChooseNumber.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ChooseNumber.test.tsx index 6c7d29eeb49..611c0e124fc 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ChooseNumber.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ChooseNumber.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { it, describe, beforeEach, vi, expect } from 'vitest' import { fireEvent, screen } from '@testing-library/react' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/EmptyFile.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/EmptyFile.test.tsx index 0a48ea73f5a..386667d8430 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/EmptyFile.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/EmptyFile.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { describe, it, expect } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ProtocolSetupParameters.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ProtocolSetupParameters.test.tsx index 0906ae6ae72..18a01391711 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ProtocolSetupParameters.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ProtocolSetupParameters.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { it, describe, beforeEach, vi, expect } from 'vitest' import { fireEvent, screen } from '@testing-library/react' @@ -31,7 +31,7 @@ vi.mock('../ChooseCsvFile') vi.mock('/app/redux/config') vi.mock('/app/organisms/ToasterOven') vi.mock('@opentrons/react-api-client') -vi.mock('/app/organisms/LabwarePositionCheck/useMostRecentCompletedAnalysis') +vi.mock('/app/resources/runs') vi.mock('react-router-dom', async importOriginal => { const reactRouterDom = await importOriginal() return { diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ResetValuesModal.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ResetValuesModal.test.tsx index 244fa6808ed..4e263f9984b 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ResetValuesModal.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ResetValuesModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, beforeEach, expect } from 'vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ViewOnlyParameters.test.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ViewOnlyParameters.test.tsx index b360ed71251..aed74fea585 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ViewOnlyParameters.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupParameters/__tests__/ViewOnlyParameters.test.tsx @@ -1,15 +1,15 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { it, describe, beforeEach, vi, expect } from 'vitest' import { fireEvent, screen } from '@testing-library/react' import { i18n } from '/app/i18n' import { renderWithProviders } from '/app/__testing-utils__' -import { useMostRecentCompletedAnalysis } from '/app/organisms/LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { useMostRecentCompletedAnalysis } from '/app/resources/runs' import { useToaster } from '/app/organisms/ToasterOven' import { mockRunTimeParameterData } from '../../__fixtures__' import { ViewOnlyParameters } from '../ViewOnlyParameters' -vi.mock('/app/organisms/LabwarePositionCheck/useMostRecentCompletedAnalysis') +vi.mock('/app/resources/runs') vi.mock('/app/organisms/ToasterOven') const RUN_ID = 'mockId' const render = (props: React.ComponentProps) => { diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupSkeleton.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupSkeleton.tsx index 4fc048a4422..55a996ef3f3 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupSkeleton.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupSkeleton.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - import { BORDERS } from '@opentrons/components' import { Skeleton } from '/app/atoms/Skeleton' diff --git a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupStep/index.tsx b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupStep/index.tsx index 5078d53b861..172aa5c7ebd 100644 --- a/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupStep/index.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/ProtocolSetupStep/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { Btn, diff --git a/app/src/organisms/ODD/ProtocolSetup/__tests__/ProtocolSetupSkeleton.test.tsx b/app/src/organisms/ODD/ProtocolSetup/__tests__/ProtocolSetupSkeleton.test.tsx index 53f5506c4a7..9fd0d8c3fa0 100644 --- a/app/src/organisms/ODD/ProtocolSetup/__tests__/ProtocolSetupSkeleton.test.tsx +++ b/app/src/organisms/ODD/ProtocolSetup/__tests__/ProtocolSetupSkeleton.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { render, screen } from '@testing-library/react' import { describe, expect, it } from 'vitest' diff --git a/app/src/organisms/ODD/RobotDashboard/EmptyRecentRun.tsx b/app/src/organisms/ODD/RobotDashboard/EmptyRecentRun.tsx index ddf462e570a..0dc551aff7a 100644 --- a/app/src/organisms/ODD/RobotDashboard/EmptyRecentRun.tsx +++ b/app/src/organisms/ODD/RobotDashboard/EmptyRecentRun.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/ODD/RobotDashboard/RecentRunProtocolCard.tsx b/app/src/organisms/ODD/RobotDashboard/RecentRunProtocolCard.tsx index 26d03fed45d..17ef7de2c3c 100644 --- a/app/src/organisms/ODD/RobotDashboard/RecentRunProtocolCard.tsx +++ b/app/src/organisms/ODD/RobotDashboard/RecentRunProtocolCard.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { css } from 'styled-components' import { useTranslation } from 'react-i18next' import { useNavigate } from 'react-router-dom' @@ -90,7 +90,7 @@ export function ProtocolWithLastRun({ // TODO(BC, 08/29/23): reintroduce this analytics event when we refactor the hook to fetch data lazily (performance concern) // const { trackProtocolRunEvent } = useTrackProtocolRunEvent(runData.id) const { cloneRun } = useCloneRun(runData.id) - const [showSpinner, setShowSpinner] = React.useState(false) + const [showSpinner, setShowSpinner] = useState(false) const protocolName = protocolData.metadata.protocolName ?? protocolData.files[0].name diff --git a/app/src/organisms/ODD/RobotDashboard/RecentRunProtocolCarousel.tsx b/app/src/organisms/ODD/RobotDashboard/RecentRunProtocolCarousel.tsx index ac9006dc93d..037347146f0 100644 --- a/app/src/organisms/ODD/RobotDashboard/RecentRunProtocolCarousel.tsx +++ b/app/src/organisms/ODD/RobotDashboard/RecentRunProtocolCarousel.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { ALIGN_FLEX_START, DIRECTION_ROW, diff --git a/app/src/organisms/ODD/RobotDashboard/ServerInitializing.tsx b/app/src/organisms/ODD/RobotDashboard/ServerInitializing.tsx index fc363f22b02..a062628a3bc 100644 --- a/app/src/organisms/ODD/RobotDashboard/ServerInitializing.tsx +++ b/app/src/organisms/ODD/RobotDashboard/ServerInitializing.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/ODD/RobotDashboard/__tests__/EmptyRecentRun.test.tsx b/app/src/organisms/ODD/RobotDashboard/__tests__/EmptyRecentRun.test.tsx index 5cfd0b070b6..6c7c7e5371c 100644 --- a/app/src/organisms/ODD/RobotDashboard/__tests__/EmptyRecentRun.test.tsx +++ b/app/src/organisms/ODD/RobotDashboard/__tests__/EmptyRecentRun.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { describe, expect, it } from 'vitest' diff --git a/app/src/organisms/ODD/RobotDashboard/__tests__/RecentRunProtocolCard.test.tsx b/app/src/organisms/ODD/RobotDashboard/__tests__/RecentRunProtocolCard.test.tsx index e3bdb0fd63c..e03f548f069 100644 --- a/app/src/organisms/ODD/RobotDashboard/__tests__/RecentRunProtocolCard.test.tsx +++ b/app/src/organisms/ODD/RobotDashboard/__tests__/RecentRunProtocolCard.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { formatDistance } from 'date-fns' import { MemoryRouter } from 'react-router-dom' diff --git a/app/src/organisms/ODD/RobotDashboard/__tests__/RecentRunProtocolCarousel.test.tsx b/app/src/organisms/ODD/RobotDashboard/__tests__/RecentRunProtocolCarousel.test.tsx index be90caa09e4..277edf80d87 100644 --- a/app/src/organisms/ODD/RobotDashboard/__tests__/RecentRunProtocolCarousel.test.tsx +++ b/app/src/organisms/ODD/RobotDashboard/__tests__/RecentRunProtocolCarousel.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { beforeEach, describe, it, vi } from 'vitest' diff --git a/app/src/organisms/ODD/RobotDashboard/hooks/__tests__/useHardwareStatusText.test.tsx b/app/src/organisms/ODD/RobotDashboard/hooks/__tests__/useHardwareStatusText.test.tsx index 04388bda536..f209e69f5ec 100644 --- a/app/src/organisms/ODD/RobotDashboard/hooks/__tests__/useHardwareStatusText.test.tsx +++ b/app/src/organisms/ODD/RobotDashboard/hooks/__tests__/useHardwareStatusText.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { I18nextProvider } from 'react-i18next' import { renderHook } from '@testing-library/react' import { beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/DeviceReset.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/DeviceReset.tsx index d7416bbe625..ab3349b62b7 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/DeviceReset.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/DeviceReset.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect, Fragment } from 'react' import { useDispatch, useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' import styled from 'styled-components' @@ -57,7 +57,7 @@ export function DeviceReset({ setCurrentOption, }: DeviceResetProps): JSX.Element { const { t } = useTranslation('device_settings') - const [resetOptions, setResetOptions] = React.useState({}) + const [resetOptions, setResetOptions] = useState({}) const options = useSelector((state: State) => getResetConfigOptions(state, robotName) ) @@ -141,11 +141,11 @@ export function DeviceReset({ subText, } } - React.useEffect(() => { + useEffect(() => { dispatch(fetchResetConfigOptions(robotName)) }, [dispatch, robotName]) - React.useEffect(() => { + useEffect(() => { if ( isEveryOptionSelected(resetOptions) && (!resetOptions.authorizedKeys || @@ -161,7 +161,7 @@ export function DeviceReset({ } }, [resetOptions]) - React.useEffect(() => { + useEffect(() => { if ( !isEveryOptionSelected(resetOptions) && resetOptions.authorizedKeys && @@ -205,7 +205,7 @@ export function DeviceReset({ {availableOptionsToDisplay.map(option => { const { optionText, subText } = renderText(option.id) return ( - + - + ) })} diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/EthernetConnectionDetails.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/EthernetConnectionDetails.tsx index 34653237faa..ddf80031dc2 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/EthernetConnectionDetails.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/EthernetConnectionDetails.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' import { css } from 'styled-components' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/NetworkDetailsModal.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/NetworkDetailsModal.tsx index a6b6c5fa473..ef835573c8f 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/NetworkDetailsModal.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/NetworkDetailsModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/RobotSettingsSelectAuthenticationType.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/RobotSettingsSelectAuthenticationType.tsx index bb870f8fb77..0507cd5bec5 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/RobotSettingsSelectAuthenticationType.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/RobotSettingsSelectAuthenticationType.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { DIRECTION_COLUMN, Flex } from '@opentrons/components' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/RobotSettingsSetWifiCred.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/RobotSettingsSetWifiCred.tsx index f86e4d977b1..0b06b66fb2b 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/RobotSettingsSetWifiCred.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/RobotSettingsSetWifiCred.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { DIRECTION_COLUMN, Flex } from '@opentrons/components' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/RobotSettingsWifi.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/RobotSettingsWifi.tsx index 081c9967b38..0c78423cca9 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/RobotSettingsWifi.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/RobotSettingsWifi.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { DIRECTION_COLUMN, Flex } from '@opentrons/components' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/RobotSettingsWifiConnect.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/RobotSettingsWifiConnect.tsx index 70377aef3c8..3fc57e9028d 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/RobotSettingsWifiConnect.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/RobotSettingsWifiConnect.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { Flex, DIRECTION_COLUMN, SPACING } from '@opentrons/components' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/WifiConnectionDetails.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/WifiConnectionDetails.tsx index 0d94ec04d55..cc980592572 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/WifiConnectionDetails.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/WifiConnectionDetails.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' @@ -42,10 +42,9 @@ export function WifiConnectionDetails({ handleJoinAnotherNetwork, }: WifiConnectionDetailsProps): JSX.Element { const { i18n, t } = useTranslation(['device_settings', 'shared']) - const [ - showNetworkDetailModal, - setShowNetworkDetailModal, - ] = React.useState(false) + const [showNetworkDetailModal, setShowNetworkDetailModal] = useState( + false + ) const localRobot = useSelector(getLocalRobot) const robotName = localRobot?.name != null ? localRobot.name : 'no name' const list = useWifiList(robotName, FETCH_WIFI_LIST_MS) diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/__tests__/EthernetConnectionDetails.test.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/__tests__/EthernetConnectionDetails.test.tsx index e4d1a096ec7..ddefd80196d 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/__tests__/EthernetConnectionDetails.test.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/__tests__/EthernetConnectionDetails.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/__tests__/NetworkDetailsModal.test.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/__tests__/NetworkDetailsModal.test.tsx index 21cd6fbc4c6..76b4c6f1be0 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/__tests__/NetworkDetailsModal.test.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/__tests__/NetworkDetailsModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/__tests__/NetworkSettings.test.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/__tests__/NetworkSettings.test.tsx index 5c1244ba66c..266778c0c81 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/__tests__/NetworkSettings.test.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/__tests__/NetworkSettings.test.tsx @@ -1,5 +1,5 @@ /* eslint-disable testing-library/no-node-access */ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/__tests__/WifiConnectionDetails.test.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/__tests__/WifiConnectionDetails.test.tsx index b7d6aeafe18..9650a89b76c 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/__tests__/WifiConnectionDetails.test.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/__tests__/WifiConnectionDetails.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { when } from 'vitest-when' import { describe, it, expect, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/index.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/index.tsx index f3e6c1ea351..c9daabfc820 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/index.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/NetworkSettings/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { useTranslation } from 'react-i18next' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/OnOffToggle.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/OnOffToggle.tsx index 565a8fb0798..32b3a94e474 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/OnOffToggle.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/OnOffToggle.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { ALIGN_CENTER, diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/Privacy.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/Privacy.tsx index 5954ae3850e..a07c12fe444 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/Privacy.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/Privacy.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useDispatch, useSelector } from 'react-redux' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/RobotName.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/RobotName.tsx index cb321fdaf95..3e640f2b4c0 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/RobotName.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/RobotName.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/RobotSettingButton.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/RobotSettingButton.tsx index da16402f0e2..f777c9fcb77 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/RobotSettingButton.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/RobotSettingButton.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { css } from 'styled-components' import { diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/RobotSystemVersion.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/RobotSystemVersion.tsx index 0cb3087c1f1..d3c26eb3161 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/RobotSystemVersion.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/RobotSystemVersion.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { @@ -42,7 +42,7 @@ export function RobotSystemVersion({ 'app_settings', 'branded', ]) - const [showModal, setShowModal] = React.useState(isUpdateAvailable) + const [showModal, setShowModal] = useState(isUpdateAvailable) return ( <> diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/RobotSystemVersionModal.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/RobotSystemVersionModal.tsx index d08ea8b7328..9d09a2cf6b2 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/RobotSystemVersionModal.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/RobotSystemVersionModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useNavigate } from 'react-router-dom' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/TextSize.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/TextSize.tsx index ad71888349f..56ca4f8c053 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/TextSize.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/TextSize.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import styled from 'styled-components' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/TouchscreenBrightness.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/TouchscreenBrightness.tsx index 2d2e7522fe5..830994c6924 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/TouchscreenBrightness.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/TouchscreenBrightness.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useDispatch, useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' import styled from 'styled-components' @@ -54,7 +54,7 @@ export function TouchscreenBrightness({ const { t } = useTranslation(['device_settings']) const dispatch = useDispatch() const initialBrightness = useSelector(getOnDeviceDisplaySettings).brightness - const [brightness, setBrightness] = React.useState(initialBrightness) + const [brightness, setBrightness] = useState(initialBrightness) const brightnessLevel = [6, 5, 4, 3, 2, 1] const handleClick = (changeType: 'up' | 'down'): void => { diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/DeviceReset.test.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/DeviceReset.test.tsx index 8c04c6effa2..9f71205231e 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/DeviceReset.test.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/DeviceReset.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/Privacy.test.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/Privacy.test.tsx index 82b72d2707e..03f4a987462 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/Privacy.test.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/Privacy.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, describe, beforeEach, afterEach, expect, it } from 'vitest' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/RobotSystemVersion.test.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/RobotSystemVersion.test.tsx index b5088e678c0..ad30e2539fa 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/RobotSystemVersion.test.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/RobotSystemVersion.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/RobotSystemVersionModal.test.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/RobotSystemVersionModal.test.tsx index 7ddfe109485..0d7a125eb1f 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/RobotSystemVersionModal.test.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/RobotSystemVersionModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/TextSize.test.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/TextSize.test.tsx index e58d805514c..703323c0d7e 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/TextSize.test.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/TextSize.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/TouchScreenSleep.test.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/TouchScreenSleep.test.tsx index c1cfde27b4e..990c6bcf436 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/TouchScreenSleep.test.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/TouchScreenSleep.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import { i18n } from '/app/i18n' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/TouchscreenBrightness.test.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/TouchscreenBrightness.test.tsx index 5795f7a1bba..76993c42300 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/TouchscreenBrightness.test.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/TouchscreenBrightness.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/UpdateChannel.test.tsx b/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/UpdateChannel.test.tsx index dd9b4c217be..c25e9582a4b 100644 --- a/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/UpdateChannel.test.tsx +++ b/app/src/organisms/ODD/RobotSettingsDashboard/__tests__/UpdateChannel.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/ODD/RunningProtocol/CancelingRunModal.tsx b/app/src/organisms/ODD/RunningProtocol/CancelingRunModal.tsx index ff5f096fa64..1aa7c2c306e 100644 --- a/app/src/organisms/ODD/RunningProtocol/CancelingRunModal.tsx +++ b/app/src/organisms/ODD/RunningProtocol/CancelingRunModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/ODD/RunningProtocol/ConfirmCancelRunModal.tsx b/app/src/organisms/ODD/RunningProtocol/ConfirmCancelRunModal.tsx index c0036d03953..453e3152ad4 100644 --- a/app/src/organisms/ODD/RunningProtocol/ConfirmCancelRunModal.tsx +++ b/app/src/organisms/ODD/RunningProtocol/ConfirmCancelRunModal.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { useTranslation } from 'react-i18next' import { useNavigate } from 'react-router-dom' import { useSelector } from 'react-redux' @@ -66,7 +66,7 @@ export function ConfirmCancelRunModal({ const robotName = localRobot?.name ?? '' const { trackProtocolRunEvent } = useTrackProtocolRunEvent(runId, robotName) const navigate = useNavigate() - const [isCanceling, setIsCanceling] = React.useState(false) + const [isCanceling, setIsCanceling] = useState(false) const modalHeader: OddModalHeaderBaseProps = { title: t('cancel_run_modal_heading'), @@ -84,7 +84,7 @@ export function ConfirmCancelRunModal({ }) } - React.useEffect(() => { + useEffect(() => { if (runStatus === RUN_STATUS_STOPPED) { trackProtocolRunEvent({ name: ANALYTICS_PROTOCOL_RUN_ACTION.CANCEL }) if (!isActiveRun) { diff --git a/app/src/organisms/ODD/RunningProtocol/CurrentRunningProtocolCommand.tsx b/app/src/organisms/ODD/RunningProtocol/CurrentRunningProtocolCommand.tsx index 8b2e6c9bcc1..e474d58ab11 100644 --- a/app/src/organisms/ODD/RunningProtocol/CurrentRunningProtocolCommand.tsx +++ b/app/src/organisms/ODD/RunningProtocol/CurrentRunningProtocolCommand.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css, keyframes } from 'styled-components' import { useTranslation } from 'react-i18next' @@ -21,7 +20,7 @@ import { import { RUN_STATUS_RUNNING, RUN_STATUS_IDLE } from '@opentrons/api-client' import { CommandText } from '/app/molecules/Command' -import { RunTimer } from '../../Devices/ProtocolRun/RunTimer' +import { RunTimer } from '/app/molecules/RunTimer' import { getCommandTextData } from '/app/molecules/Command/utils/getCommandTextData' import { PlayPauseButton } from './PlayPauseButton' import { StopButton } from './StopButton' diff --git a/app/src/organisms/ODD/RunningProtocol/PlayPauseButton.tsx b/app/src/organisms/ODD/RunningProtocol/PlayPauseButton.tsx index 47428f1420b..44b4154cf65 100644 --- a/app/src/organisms/ODD/RunningProtocol/PlayPauseButton.tsx +++ b/app/src/organisms/ODD/RunningProtocol/PlayPauseButton.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { diff --git a/app/src/organisms/ODD/RunningProtocol/RunFailedModal.tsx b/app/src/organisms/ODD/RunningProtocol/RunFailedModal.tsx index 1de8c5138d7..78d5e06d2d2 100644 --- a/app/src/organisms/ODD/RunningProtocol/RunFailedModal.tsx +++ b/app/src/organisms/ODD/RunningProtocol/RunFailedModal.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { useNavigate } from 'react-router-dom' import { css } from 'styled-components' @@ -46,7 +46,7 @@ export function RunFailedModal({ const { t, i18n } = useTranslation(['run_details', 'shared', 'branded']) const navigate = useNavigate() const { stopRun } = useStopRunMutation() - const [isCanceling, setIsCanceling] = React.useState(false) + const [isCanceling, setIsCanceling] = useState(false) if ( (errors == null || errors.length === 0) && diff --git a/app/src/organisms/ODD/RunningProtocol/RunningProtocolCommandList.tsx b/app/src/organisms/ODD/RunningProtocol/RunningProtocolCommandList.tsx index f351a46541f..f161fa77e05 100644 --- a/app/src/organisms/ODD/RunningProtocol/RunningProtocolCommandList.tsx +++ b/app/src/organisms/ODD/RunningProtocol/RunningProtocolCommandList.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useRef, useState, useEffect } from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' import { ViewportList } from 'react-viewport-list' @@ -102,14 +102,14 @@ export function RunningProtocolCommandList({ currentRunCommandIndex, }: RunningProtocolCommandListProps): JSX.Element { const { t } = useTranslation('run_details') - const viewPortRef = React.useRef(null) - const ref = React.useRef(null) + const viewPortRef = useRef(null) + const ref = useRef(null) const currentRunStatus = t(`status_${runStatus}`) const onStop = (): void => { if (runStatus === RUN_STATUS_RUNNING) pauseRun() setShowConfirmCancelRunModal(true) } - const [visibleRange, setVisibleRange] = React.useState({ + const [visibleRange, setVisibleRange] = useState({ lowestVisibleIndex: 0, highestVisibleIndex: 0, }) @@ -133,7 +133,7 @@ export function RunningProtocolCommandList({ } } - React.useEffect(() => { + useEffect(() => { // Note (kk:09/25/2023) Need -1 because the element of highestVisibleIndex cannot really readable // due to limited space const isCurrentCommandVisible = diff --git a/app/src/organisms/ODD/RunningProtocol/RunningProtocolSkeleton.tsx b/app/src/organisms/ODD/RunningProtocol/RunningProtocolSkeleton.tsx index 073f650530a..d6bcc91ea21 100644 --- a/app/src/organisms/ODD/RunningProtocol/RunningProtocolSkeleton.tsx +++ b/app/src/organisms/ODD/RunningProtocol/RunningProtocolSkeleton.tsx @@ -1,5 +1,3 @@ -import * as React from 'react' - import { DIRECTION_COLUMN, DIRECTION_ROW, diff --git a/app/src/organisms/ODD/RunningProtocol/StopButton.tsx b/app/src/organisms/ODD/RunningProtocol/StopButton.tsx index 2711c8b9da2..4ee9a716c81 100644 --- a/app/src/organisms/ODD/RunningProtocol/StopButton.tsx +++ b/app/src/organisms/ODD/RunningProtocol/StopButton.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { diff --git a/app/src/organisms/ODD/RunningProtocol/__tests__/CancelingRunModal.test.tsx b/app/src/organisms/ODD/RunningProtocol/__tests__/CancelingRunModal.test.tsx index 334493b8db9..5bc8efafe42 100644 --- a/app/src/organisms/ODD/RunningProtocol/__tests__/CancelingRunModal.test.tsx +++ b/app/src/organisms/ODD/RunningProtocol/__tests__/CancelingRunModal.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { describe, it } from 'vitest' diff --git a/app/src/organisms/ODD/RunningProtocol/__tests__/ConfirmCancelRunModal.test.tsx b/app/src/organisms/ODD/RunningProtocol/__tests__/ConfirmCancelRunModal.test.tsx index 55c4b3fec65..69f610d3ef8 100644 --- a/app/src/organisms/ODD/RunningProtocol/__tests__/ConfirmCancelRunModal.test.tsx +++ b/app/src/organisms/ODD/RunningProtocol/__tests__/ConfirmCancelRunModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/organisms/ODD/RunningProtocol/__tests__/CurrentRunningProtocolCommand.test.tsx b/app/src/organisms/ODD/RunningProtocol/__tests__/CurrentRunningProtocolCommand.test.tsx index 7cde75fdbaf..8e19bb69491 100644 --- a/app/src/organisms/ODD/RunningProtocol/__tests__/CurrentRunningProtocolCommand.test.tsx +++ b/app/src/organisms/ODD/RunningProtocol/__tests__/CurrentRunningProtocolCommand.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ODD/RunningProtocol/__tests__/RunFailedModal.test.tsx b/app/src/organisms/ODD/RunningProtocol/__tests__/RunFailedModal.test.tsx index 09f86423fcf..8dcfd2e5b88 100644 --- a/app/src/organisms/ODD/RunningProtocol/__tests__/RunFailedModal.test.tsx +++ b/app/src/organisms/ODD/RunningProtocol/__tests__/RunFailedModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ODD/RunningProtocol/__tests__/RunningProtocolCommandList.test.tsx b/app/src/organisms/ODD/RunningProtocol/__tests__/RunningProtocolCommandList.test.tsx index 3504d069ee5..f506dce8404 100644 --- a/app/src/organisms/ODD/RunningProtocol/__tests__/RunningProtocolCommandList.test.tsx +++ b/app/src/organisms/ODD/RunningProtocol/__tests__/RunningProtocolCommandList.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { beforeEach, describe, expect, it, vi } from 'vitest' diff --git a/app/src/organisms/ODD/RunningProtocol/__tests__/RunningProtocolSkeleton.test.tsx b/app/src/organisms/ODD/RunningProtocol/__tests__/RunningProtocolSkeleton.test.tsx index 33bc166f181..656d4d250d1 100644 --- a/app/src/organisms/ODD/RunningProtocol/__tests__/RunningProtocolSkeleton.test.tsx +++ b/app/src/organisms/ODD/RunningProtocol/__tests__/RunningProtocolSkeleton.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { beforeEach, describe, expect, it } from 'vitest' diff --git a/app/src/organisms/ODD/hooks/__tests__/useIsUnboxingFlowOngoing.test.tsx b/app/src/organisms/ODD/hooks/__tests__/useIsUnboxingFlowOngoing.test.tsx index 4f4bec2ccbe..d7610096015 100644 --- a/app/src/organisms/ODD/hooks/__tests__/useIsUnboxingFlowOngoing.test.tsx +++ b/app/src/organisms/ODD/hooks/__tests__/useIsUnboxingFlowOngoing.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { renderHook } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import { Provider } from 'react-redux' diff --git a/app/src/organisms/OpenDoorAlertModal/__tests__/OpenDoorAlertModal.test.tsx b/app/src/organisms/OpenDoorAlertModal/__tests__/OpenDoorAlertModal.test.tsx index 9bd39379845..a431ecac1cb 100644 --- a/app/src/organisms/OpenDoorAlertModal/__tests__/OpenDoorAlertModal.test.tsx +++ b/app/src/organisms/OpenDoorAlertModal/__tests__/OpenDoorAlertModal.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { describe, it } from 'vitest' diff --git a/app/src/organisms/OpenDoorAlertModal/index.tsx b/app/src/organisms/OpenDoorAlertModal/index.tsx index b12435d9c99..f9ac5440823 100644 --- a/app/src/organisms/OpenDoorAlertModal/index.tsx +++ b/app/src/organisms/OpenDoorAlertModal/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/PipetteWizardFlows/AttachProbe.tsx b/app/src/organisms/PipetteWizardFlows/AttachProbe.tsx index 24086bc6e6f..66f40287463 100644 --- a/app/src/organisms/PipetteWizardFlows/AttachProbe.tsx +++ b/app/src/organisms/PipetteWizardFlows/AttachProbe.tsx @@ -1,7 +1,8 @@ -import * as React from 'react' +import { useState } from 'react' import { css } from 'styled-components' import { Trans, useTranslation } from 'react-i18next' import { + Banner, COLORS, Flex, RESPONSIVENESS, @@ -10,7 +11,6 @@ import { TYPOGRAPHY, } from '@opentrons/components' import { LEFT, WASTE_CHUTE_CUTOUT } from '@opentrons/shared-data' -import { Banner } from '/app/atoms/Banner' import { GenericWizardTile } from '/app/molecules/GenericWizardTile' import { SimpleWizardBody, @@ -58,9 +58,7 @@ export const AttachProbe = (props: AttachProbeProps): JSX.Element | null => { } = props const { t, i18n } = useTranslation('pipette_wizard_flows') const pipetteWizardStep = { mount, flowType, section: SECTIONS.ATTACH_PROBE } - const [showUnableToDetect, setShowUnableToDetect] = React.useState( - false - ) + const [showUnableToDetect, setShowUnableToDetect] = useState(false) const pipetteId = attachedPipettes[mount]?.serialNumber const displayName = attachedPipettes[mount]?.displayName diff --git a/app/src/organisms/PipetteWizardFlows/BeforeBeginning.tsx b/app/src/organisms/PipetteWizardFlows/BeforeBeginning.tsx index f781e435f2a..ffcb72dae18 100644 --- a/app/src/organisms/PipetteWizardFlows/BeforeBeginning.tsx +++ b/app/src/organisms/PipetteWizardFlows/BeforeBeginning.tsx @@ -1,7 +1,8 @@ -import * as React from 'react' +import { useEffect } from 'react' import { Trans, useTranslation } from 'react-i18next' import { COLORS, + Banner, DIRECTION_COLUMN, Flex, SPACING, @@ -14,14 +15,13 @@ import { WEIGHT_OF_96_CHANNEL, WASTE_CHUTE_CUTOUT, } from '@opentrons/shared-data' -import { Banner } from '/app/atoms/Banner' import { SimpleWizardBody, SimpleWizardInProgressBody, } from '/app/molecules/SimpleWizardBody' import { GenericWizardTile } from '/app/molecules/GenericWizardTile' import { WizardRequiredEquipmentList } from '/app/molecules/WizardRequiredEquipmentList' -import { usePipetteNameSpecs } from '/app/resources/instruments/hooks' +import { usePipetteNameSpecs } from '/app/local-resources/instruments' import { CALIBRATION_PROBE, FLOWS, @@ -79,7 +79,7 @@ export const BeforeBeginning = ( createdMaintenanceRunId, } = props const { t } = useTranslation(['pipette_wizard_flows', 'shared']) - React.useEffect(() => { + useEffect(() => { if (createdMaintenanceRunId == null) { createMaintenanceRun({}) } diff --git a/app/src/organisms/PipetteWizardFlows/Carriage.tsx b/app/src/organisms/PipetteWizardFlows/Carriage.tsx index cdd75df4ffe..f4103cf424d 100644 --- a/app/src/organisms/PipetteWizardFlows/Carriage.tsx +++ b/app/src/organisms/PipetteWizardFlows/Carriage.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import capitalize from 'lodash/capitalize' import { diff --git a/app/src/organisms/PipetteWizardFlows/CheckPipetteButton.tsx b/app/src/organisms/PipetteWizardFlows/CheckPipetteButton.tsx index 83c5b1507fd..2300204b65b 100644 --- a/app/src/organisms/PipetteWizardFlows/CheckPipetteButton.tsx +++ b/app/src/organisms/PipetteWizardFlows/CheckPipetteButton.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useInstrumentsQuery } from '@opentrons/react-api-client' import { PrimaryButton } from '@opentrons/components' import { SmallButton } from '/app/atoms/buttons' diff --git a/app/src/organisms/PipetteWizardFlows/DetachPipette.tsx b/app/src/organisms/PipetteWizardFlows/DetachPipette.tsx index 8d5db88c879..46e5f92e389 100644 --- a/app/src/organisms/PipetteWizardFlows/DetachPipette.tsx +++ b/app/src/organisms/PipetteWizardFlows/DetachPipette.tsx @@ -6,18 +6,18 @@ import { useInstrumentsQuery } from '@opentrons/react-api-client' import { ALIGN_CENTER, ALIGN_FLEX_END, + Banner, Btn, COLORS, Flex, JUSTIFY_SPACE_BETWEEN, + LegacyStyledText, PrimaryButton, RESPONSIVENESS, SIZE_1, SPACING, - LegacyStyledText, TYPOGRAPHY, } from '@opentrons/components' -import { Banner } from '/app/atoms/Banner' import { GenericWizardTile } from '/app/molecules/GenericWizardTile' import { SimpleWizardBody, diff --git a/app/src/organisms/PipetteWizardFlows/DetachProbe.tsx b/app/src/organisms/PipetteWizardFlows/DetachProbe.tsx index 03603bd640c..833f28d45dc 100644 --- a/app/src/organisms/PipetteWizardFlows/DetachProbe.tsx +++ b/app/src/organisms/PipetteWizardFlows/DetachProbe.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { LegacyStyledText } from '@opentrons/components' import { GenericWizardTile } from '/app/molecules/GenericWizardTile' diff --git a/app/src/organisms/PipetteWizardFlows/ExitModal.tsx b/app/src/organisms/PipetteWizardFlows/ExitModal.tsx index 90430769d8f..d2bc3517643 100644 --- a/app/src/organisms/PipetteWizardFlows/ExitModal.tsx +++ b/app/src/organisms/PipetteWizardFlows/ExitModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import capitalize from 'lodash/capitalize' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/PipetteWizardFlows/MountPipette.tsx b/app/src/organisms/PipetteWizardFlows/MountPipette.tsx index 6234d5247a0..9ba1b036785 100644 --- a/app/src/organisms/PipetteWizardFlows/MountPipette.tsx +++ b/app/src/organisms/PipetteWizardFlows/MountPipette.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { SINGLE_MOUNT_PIPETTES, @@ -6,12 +6,12 @@ import { } from '@opentrons/shared-data' import { Flex, + Banner, JUSTIFY_CENTER, SIZE_1, SPACING, LegacyStyledText, } from '@opentrons/components' -import { Banner } from '/app/atoms/Banner' import { GenericWizardTile } from '/app/molecules/GenericWizardTile' import { Skeleton } from '/app/atoms/Skeleton' import { CheckPipetteButton } from './CheckPipetteButton' diff --git a/app/src/organisms/PipetteWizardFlows/MountingPlate.tsx b/app/src/organisms/PipetteWizardFlows/MountingPlate.tsx index a09ee3d2c1b..aaa6178fc79 100644 --- a/app/src/organisms/PipetteWizardFlows/MountingPlate.tsx +++ b/app/src/organisms/PipetteWizardFlows/MountingPlate.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { Trans, useTranslation } from 'react-i18next' import { LEFT } from '@opentrons/shared-data' import { COLORS, SPACING, LegacyStyledText } from '@opentrons/components' diff --git a/app/src/organisms/PipetteWizardFlows/ProbeNotAttached.tsx b/app/src/organisms/PipetteWizardFlows/ProbeNotAttached.tsx index d959065ac49..d98fde280f7 100644 --- a/app/src/organisms/PipetteWizardFlows/ProbeNotAttached.tsx +++ b/app/src/organisms/PipetteWizardFlows/ProbeNotAttached.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { ALIGN_CENTER, @@ -32,7 +32,7 @@ export const ProbeNotAttached = ( 'branded', ]) const { isOnDevice, handleOnClick, setShowUnableToDetect } = props - const [numberOfTryAgains, setNumberOfTryAgains] = React.useState(0) + const [numberOfTryAgains, setNumberOfTryAgains] = useState(0) return ( props.pipetteInfo ?? null, []) - const isGantryEmpty = React.useMemo( + const memoizedPipetteInfo = useMemo(() => props.pipetteInfo ?? null, []) + const isGantryEmpty = useMemo( () => attachedPipettes[LEFT] == null && attachedPipettes[RIGHT] == null, [] ) - const pipetteWizardSteps = React.useMemo( + const pipetteWizardSteps = useMemo( () => memoizedPipetteInfo == null ? getPipetteWizardSteps(flowType, mount, selectedPipette, isGantryEmpty) @@ -91,14 +91,12 @@ export const PipetteWizardFlows = ( pipette => pipette.mount === mount ) const host = useHost() - const [currentStepIndex, setCurrentStepIndex] = React.useState(0) + const [currentStepIndex, setCurrentStepIndex] = useState(0) const totalStepCount = pipetteWizardSteps != null ? pipetteWizardSteps.length - 1 : 0 const currentStep = pipetteWizardSteps?.[currentStepIndex] ?? null - const [isFetchingPipettes, setIsFetchingPipettes] = React.useState( - false - ) - const memoizedAttachedPipettes = React.useMemo(() => attachedPipettes, []) + const [isFetchingPipettes, setIsFetchingPipettes] = useState(false) + const memoizedAttachedPipettes = useMemo(() => attachedPipettes, []) const hasCalData = memoizedAttachedPipettes[mount]?.data.calibratedOffset?.last_modified != null @@ -111,8 +109,8 @@ export const PipetteWizardFlows = ( attachedPipettes: memoizedAttachedPipettes, pipetteInfo: memoizedPipetteInfo, }) - const memoizedWizardTitle = React.useMemo(() => wizardTitle, []) - const [createdMaintenanceRunId, setCreatedMaintenanceRunId] = React.useState< + const memoizedWizardTitle = useMemo(() => wizardTitle, []) + const [createdMaintenanceRunId, setCreatedMaintenanceRunId] = useState< string | null >(null) // we should start checking for run deletion only after the maintenance run is created @@ -120,7 +118,7 @@ export const PipetteWizardFlows = ( const [ monitorMaintenanceRunForDeletion, setMonitorMaintenanceRunForDeletion, - ] = React.useState(false) + ] = useState(false) const goBack = (): void => { setCurrentStepIndex( @@ -151,7 +149,7 @@ export const PipetteWizardFlows = ( // this will close the modal in case the run was deleted by the terminate // activity modal on the ODD - React.useEffect(() => { + useEffect(() => { if ( createdMaintenanceRunId !== null && maintenanceRunData?.data.id === createdMaintenanceRunId @@ -171,10 +169,8 @@ export const PipetteWizardFlows = ( closeFlow, ]) - const [errorMessage, setShowErrorMessage] = React.useState( - null - ) - const [isExiting, setIsExiting] = React.useState(false) + const [errorMessage, setShowErrorMessage] = useState(null) + const [isExiting, setIsExiting] = useState(false) const proceed = (): void => { if (!isCommandMutationLoading) { setCurrentStepIndex( @@ -185,12 +181,13 @@ export const PipetteWizardFlows = ( } } const handleClose = (): void => { - if (onComplete != null) onComplete() + if (onComplete != null) { + onComplete() + } if (maintenanceRunData != null) { deleteMaintenanceRun(maintenanceRunData?.data.id) - } else { - closeFlow() } + closeFlow() } const { @@ -213,13 +210,13 @@ export const PipetteWizardFlows = ( [{ commandType: 'home' as const, params: {} }], false ) - .then(() => { - handleClose() - }) .catch(error => { setIsExiting(true) setShowErrorMessage(error.message as string) }) + .finally(() => { + handleClose() + }) } } const { diff --git a/app/src/organisms/PipetteWizardFlows/utils.tsx b/app/src/organisms/PipetteWizardFlows/utils.tsx index 1eb2ad83632..c4be3be880d 100644 --- a/app/src/organisms/PipetteWizardFlows/utils.tsx +++ b/app/src/organisms/PipetteWizardFlows/utils.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { LEFT, RIGHT } from '@opentrons/shared-data' import { SPACING } from '@opentrons/components' diff --git a/app/src/organisms/ProtocolAnalysisFailure/ProtocolAnalysisStale.tsx b/app/src/organisms/ProtocolAnalysisFailure/ProtocolAnalysisStale.tsx index aa961fae559..9b3a7c00d14 100644 --- a/app/src/organisms/ProtocolAnalysisFailure/ProtocolAnalysisStale.tsx +++ b/app/src/organisms/ProtocolAnalysisFailure/ProtocolAnalysisStale.tsx @@ -1,9 +1,10 @@ -import * as React from 'react' +import type * as React from 'react' import { useDispatch } from 'react-redux' import { useTranslation, Trans } from 'react-i18next' import { ALIGN_CENTER, + Banner, Btn, Flex, JUSTIFY_SPACE_BETWEEN, @@ -13,8 +14,6 @@ import { WRAP_REVERSE, } from '@opentrons/components' -import { Banner } from '/app/atoms/Banner' - import type { Dispatch } from '/app/redux/types' import { analyzeProtocol } from '/app/redux/protocol-storage' interface ProtocolAnalysisStaleProps { diff --git a/app/src/organisms/ProtocolAnalysisFailure/__tests__/ProtocolAnalysisFailure.test.tsx b/app/src/organisms/ProtocolAnalysisFailure/__tests__/ProtocolAnalysisFailure.test.tsx index ff8d5fa225d..fbdec0a45ec 100644 --- a/app/src/organisms/ProtocolAnalysisFailure/__tests__/ProtocolAnalysisFailure.test.tsx +++ b/app/src/organisms/ProtocolAnalysisFailure/__tests__/ProtocolAnalysisFailure.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect } from 'vitest' diff --git a/app/src/organisms/ProtocolAnalysisFailure/index.tsx b/app/src/organisms/ProtocolAnalysisFailure/index.tsx index a5d35f32a42..c16f63b3838 100644 --- a/app/src/organisms/ProtocolAnalysisFailure/index.tsx +++ b/app/src/organisms/ProtocolAnalysisFailure/index.tsx @@ -7,6 +7,7 @@ import { css } from 'styled-components' import { ALIGN_CENTER, Btn, + Banner, Flex, JUSTIFY_FLEX_END, Modal, @@ -19,7 +20,6 @@ import { } from '@opentrons/components' import { analyzeProtocol } from '/app/redux/protocol-storage' -import { Banner } from '/app/atoms/Banner' import { getTopPortalEl } from '../../App/portal' import type { Dispatch } from '/app/redux/types' diff --git a/app/src/organisms/ProtocolDetails/AnnotatedSteps.tsx b/app/src/organisms/ProtocolDetails/AnnotatedSteps.tsx index 4e15ad14b62..bdc5848f03b 100644 --- a/app/src/organisms/ProtocolDetails/AnnotatedSteps.tsx +++ b/app/src/organisms/ProtocolDetails/AnnotatedSteps.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { FLEX_ROBOT_TYPE } from '@opentrons/shared-data' import { diff --git a/app/src/organisms/ProtocolDetails/ProtocolLiquidsDetails.tsx b/app/src/organisms/ProtocolDetails/ProtocolLiquidsDetails.tsx index 0cb9780dee7..2c255666fcd 100644 --- a/app/src/organisms/ProtocolDetails/ProtocolLiquidsDetails.tsx +++ b/app/src/organisms/ProtocolDetails/ProtocolLiquidsDetails.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { Fragment } from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' @@ -50,7 +50,7 @@ export const ProtocolLiquidsDetails = ( {liquidsInLoadOrder.length > 0 ? ( liquidsInLoadOrder?.map((liquid, index) => { return ( - + {index < liquidsInLoadOrder.length - 1 && } - + ) }) ) : ( diff --git a/app/src/organisms/ProtocolDetails/ProtocolParameters/__tests__/ProtocolParameters.test.tsx b/app/src/organisms/ProtocolDetails/ProtocolParameters/__tests__/ProtocolParameters.test.tsx index e87c660e0d9..e52803cc054 100644 --- a/app/src/organisms/ProtocolDetails/ProtocolParameters/__tests__/ProtocolParameters.test.tsx +++ b/app/src/organisms/ProtocolDetails/ProtocolParameters/__tests__/ProtocolParameters.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, vi, beforeEach, afterEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/ProtocolDetails/ProtocolParameters/index.tsx b/app/src/organisms/ProtocolDetails/ProtocolParameters/index.tsx index 6e7067204d0..23ee98c1649 100644 --- a/app/src/organisms/ProtocolDetails/ProtocolParameters/index.tsx +++ b/app/src/organisms/ProtocolDetails/ProtocolParameters/index.tsx @@ -1,7 +1,7 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { + Banner, DIRECTION_COLUMN, Flex, InfoScreen, @@ -9,7 +9,6 @@ import { SPACING, StyledText, } from '@opentrons/components' -import { Banner } from '/app/atoms/Banner' import type { RunTimeParameter } from '@opentrons/shared-data' diff --git a/app/src/organisms/ProtocolDetails/ProtocolStats.tsx b/app/src/organisms/ProtocolDetails/ProtocolStats.tsx index 935636690c7..589eeb2e2ea 100644 --- a/app/src/organisms/ProtocolDetails/ProtocolStats.tsx +++ b/app/src/organisms/ProtocolDetails/ProtocolStats.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' import { diff --git a/app/src/organisms/ProtocolDetails/RobotConfigurationDetails.tsx b/app/src/organisms/ProtocolDetails/RobotConfigurationDetails.tsx index d2fc14e63eb..5789a8af6e6 100644 --- a/app/src/organisms/ProtocolDetails/RobotConfigurationDetails.tsx +++ b/app/src/organisms/ProtocolDetails/RobotConfigurationDetails.tsx @@ -1,4 +1,4 @@ -import React from 'react' +import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/ProtocolDetails/__tests__/ProtocolDetails.test.tsx b/app/src/organisms/ProtocolDetails/__tests__/ProtocolDetails.test.tsx index faf7cd40053..e1884919d55 100644 --- a/app/src/organisms/ProtocolDetails/__tests__/ProtocolDetails.test.tsx +++ b/app/src/organisms/ProtocolDetails/__tests__/ProtocolDetails.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { act, screen, waitFor } from '@testing-library/react' import { MemoryRouter } from 'react-router-dom' import { describe, it, beforeEach, vi, expect, afterEach } from 'vitest' diff --git a/app/src/organisms/ProtocolDetails/__tests__/ProtocolLabwareDetails.test.tsx b/app/src/organisms/ProtocolDetails/__tests__/ProtocolLabwareDetails.test.tsx index 3d22f3d813e..93c215643cf 100644 --- a/app/src/organisms/ProtocolDetails/__tests__/ProtocolLabwareDetails.test.tsx +++ b/app/src/organisms/ProtocolDetails/__tests__/ProtocolLabwareDetails.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, beforeEach, vi } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/ProtocolDetails/__tests__/ProtocolLiquidsDetails.test.tsx b/app/src/organisms/ProtocolDetails/__tests__/ProtocolLiquidsDetails.test.tsx index eeb761277b1..f21ea09c406 100644 --- a/app/src/organisms/ProtocolDetails/__tests__/ProtocolLiquidsDetails.test.tsx +++ b/app/src/organisms/ProtocolDetails/__tests__/ProtocolLiquidsDetails.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, beforeEach, vi } from 'vitest' import { parseLiquidsInLoadOrder } from '@opentrons/shared-data' diff --git a/app/src/organisms/ProtocolDetails/__tests__/RobotConfigurationDetails.test.tsx b/app/src/organisms/ProtocolDetails/__tests__/RobotConfigurationDetails.test.tsx index e7f473c837d..b823732ce95 100644 --- a/app/src/organisms/ProtocolDetails/__tests__/RobotConfigurationDetails.test.tsx +++ b/app/src/organisms/ProtocolDetails/__tests__/RobotConfigurationDetails.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, afterEach, vi } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/ProtocolDetails/index.tsx b/app/src/organisms/ProtocolDetails/index.tsx index 5df7bd33e4a..213c20d3074 100644 --- a/app/src/organisms/ProtocolDetails/index.tsx +++ b/app/src/organisms/ProtocolDetails/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, Fragment } from 'react' import { createPortal } from 'react-dom' import map from 'lodash/map' import omit from 'lodash/omit' @@ -142,7 +142,7 @@ function MetadataDetails({ {filteredMetaData.map((item, index) => { return ( - + {item.value} - + ) })} @@ -167,7 +167,7 @@ interface ReadMoreContentProps { const ReadMoreContent = (props: ReadMoreContentProps): JSX.Element => { const { metadata, protocolType } = props const { t, i18n } = useTranslation('protocol_details') - const [isReadMore, setIsReadMore] = React.useState(true) + const [isReadMore, setIsReadMore] = useState(true) const description = isEmpty(metadata.description) ? t('shared:no_data') @@ -217,18 +217,18 @@ export function ProtocolDetails( const enableProtocolTimeline = useFeatureFlag('protocolTimeline') const runTimeParameters = mostRecentAnalysis?.runTimeParameters ?? [] const hasRunTimeParameters = runTimeParameters.length > 0 - const [currentTab, setCurrentTab] = React.useState< + const [currentTab, setCurrentTab] = useState< 'robot_config' | 'labware' | 'liquids' | 'stats' | 'parameters' | 'timeline' >(hasRunTimeParameters ? 'parameters' : 'robot_config') const [ showChooseRobotToRunProtocolSlideout, setShowChooseRobotToRunProtocolSlideout, - ] = React.useState(false) + ] = useState(false) const [ showSendProtocolToFlexSlideout, setShowSendProtocolToFlexSlideout, - ] = React.useState(false) - const [showDeckViewModal, setShowDeckViewModal] = React.useState(false) + ] = useState(false) + const [showDeckViewModal, setShowDeckViewModal] = useState(false) const isAnalyzing = useSelector((state: State) => getIsProtocolAnalysisInProgress(state, protocolKey) diff --git a/app/src/organisms/ProtocolStatusBanner/__tests__/ProtocolStatusBanner.test.tsx b/app/src/organisms/ProtocolStatusBanner/__tests__/ProtocolStatusBanner.test.tsx index a7de3ba7ef1..3a0326c5273 100644 --- a/app/src/organisms/ProtocolStatusBanner/__tests__/ProtocolStatusBanner.test.tsx +++ b/app/src/organisms/ProtocolStatusBanner/__tests__/ProtocolStatusBanner.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, expect } from 'vitest' diff --git a/app/src/organisms/ProtocolStatusBanner/index.tsx b/app/src/organisms/ProtocolStatusBanner/index.tsx index 6037d88596a..133cbf35896 100644 --- a/app/src/organisms/ProtocolStatusBanner/index.tsx +++ b/app/src/organisms/ProtocolStatusBanner/index.tsx @@ -1,7 +1,5 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' -import { SPACING, LegacyStyledText } from '@opentrons/components' -import { Banner } from '/app/atoms/Banner' +import { SPACING, Banner, LegacyStyledText } from '@opentrons/components' import type { IconProps } from '@opentrons/components' diff --git a/app/src/organisms/ProtocolTimelineScrubber/CommandItem.tsx b/app/src/organisms/ProtocolTimelineScrubber/CommandItem.tsx index fee6d977cf7..ae9377a19a0 100644 --- a/app/src/organisms/ProtocolTimelineScrubber/CommandItem.tsx +++ b/app/src/organisms/ProtocolTimelineScrubber/CommandItem.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { Flex, DIRECTION_COLUMN, @@ -28,7 +28,7 @@ interface CommandItemProps { robotType: RobotType } export function CommandItem(props: CommandItemProps): JSX.Element { - const [showDetails, setShowDetails] = React.useState(false) + const [showDetails, setShowDetails] = useState(false) const { index, command, diff --git a/app/src/organisms/ProtocolTimelineScrubber/PipetteVisuals.tsx b/app/src/organisms/ProtocolTimelineScrubber/PipetteVisuals.tsx index 6a9ee8dc566..10224a08e49 100644 --- a/app/src/organisms/ProtocolTimelineScrubber/PipetteVisuals.tsx +++ b/app/src/organisms/ProtocolTimelineScrubber/PipetteVisuals.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { Flex, Box, @@ -32,7 +32,7 @@ export function PipetteMountViz( props: PipetteMountVizProps ): JSX.Element | null { const { mount, pipetteEntity, pipetteId, timelineFrame, analysis } = props - const [showPipetteDetails, setShowPipetteDetails] = React.useState(false) + const [showPipetteDetails, setShowPipetteDetails] = useState(false) return pipetteEntity == null ? null : ( (false) let error: string | null = null if (name.length > 60) { @@ -40,9 +41,10 @@ export function NameQuickTransfer(props: NameQuickTransferProps): JSX.Element { header={t('name_your_transfer')} buttonText={t('save')} onClickButton={() => { + setIsSaving(true) onSave(name) }} - buttonIsDisabled={name === '' || error != null} + buttonIsDisabled={name === '' || error != null || isSaving} /> ( @@ -81,10 +84,22 @@ export function AirGap(props: AirGapProps): JSX.Element { setCurrentStep(currentStep + 1) } else { dispatch({ type: action, volume: undefined }) + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_SETTING_SAVED, + properties: { + setting: `AirGap_${kind}`, + }, + }) onBack() } } else if (currentStep === 2) { dispatch({ type: action, volume: volume ?? undefined }) + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_SETTING_SAVED, + properties: { + setting: `AirGap_${kind}`, + }, + }) onBack() } } @@ -207,7 +222,7 @@ export function AirGap(props: AirGapProps): JSX.Element { > { setVolume(Number(e)) }} diff --git a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/BlowOut.tsx b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/BlowOut.tsx index 34073d18b8e..ea402bc9c12 100644 --- a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/BlowOut.tsx +++ b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/BlowOut.tsx @@ -15,8 +15,10 @@ import { FLEX_SINGLE_SLOT_BY_CUTOUT_ID, TRASH_BIN_ADAPTER_FIXTURE, } from '@opentrons/shared-data' +import { ANALYTICS_QUICK_TRANSFER_SETTING_SAVED } from '../../../redux/analytics' import { getTopPortalEl } from '../../../App/portal' import { useNotifyDeckConfigurationQuery } from '/app/resources/deck_configuration' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' import { ChildNavigation } from '../../ChildNavigation' import { ACTIONS } from '../constants' @@ -91,6 +93,7 @@ export const useBlowOutLocationOptions = ( export function BlowOut(props: BlowOutProps): JSX.Element { const { onBack, state, dispatch } = props const { t } = useTranslation('quick_transfer') + const { trackEventWithRobotSerial } = useTrackEventWithRobotSerial() const deckConfig = useNotifyDeckConfigurationQuery().data ?? [] const [isBlowOutEnabled, setisBlowOutEnabled] = React.useState( @@ -134,6 +137,12 @@ export function BlowOut(props: BlowOutProps): JSX.Element { type: ACTIONS.SET_BLOW_OUT, location: undefined, }) + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_SETTING_SAVED, + properties: { + settting: `BlowOut`, + }, + }) onBack() } else { setCurrentStep(currentStep + 1) @@ -143,6 +152,12 @@ export function BlowOut(props: BlowOutProps): JSX.Element { type: ACTIONS.SET_BLOW_OUT, location: blowOutLocation, }) + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_SETTING_SAVED, + properties: { + settting: `BlowOut`, + }, + }) onBack() } } diff --git a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/Delay.tsx b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/Delay.tsx index 75e2389bf5d..61fc8f3cc10 100644 --- a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/Delay.tsx +++ b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/Delay.tsx @@ -13,8 +13,10 @@ import { SPACING, } from '@opentrons/components' -import { getTopPortalEl } from '../../../App/portal' +import { ANALYTICS_QUICK_TRANSFER_SETTING_SAVED } from '/app/redux/analytics' +import { getTopPortalEl } from '/app/App/portal' import { ChildNavigation } from '../../ChildNavigation' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' import { ACTIONS } from '../constants' import type { @@ -35,6 +37,7 @@ interface DelayProps { export function Delay(props: DelayProps): JSX.Element { const { kind, onBack, state, dispatch } = props const { t } = useTranslation('quick_transfer') + const { trackEventWithRobotSerial } = useTrackEventWithRobotSerial() const keyboardRef = React.useRef(null) const [currentStep, setCurrentStep] = React.useState(1) @@ -87,6 +90,12 @@ export function Delay(props: DelayProps): JSX.Element { type: action, delaySettings: undefined, }) + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_SETTING_SAVED, + properties: { + settting: `Delay_${kind}`, + }, + }) onBack() } else { setCurrentStep(2) @@ -102,6 +111,12 @@ export function Delay(props: DelayProps): JSX.Element { positionFromBottom: position, }, }) + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_SETTING_SAVED, + properties: { + settting: `Delay_${kind}`, + }, + }) } onBack() } @@ -224,6 +239,7 @@ export function Delay(props: DelayProps): JSX.Element { > { setDelayDuration(Number(e)) }} @@ -264,7 +280,7 @@ export function Delay(props: DelayProps): JSX.Element { > { setPosition(Number(e)) }} diff --git a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/FlowRate.tsx b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/FlowRate.tsx index dbbad16c11c..993ee01ca32 100644 --- a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/FlowRate.tsx +++ b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/FlowRate.tsx @@ -15,10 +15,13 @@ import { LOW_VOLUME_PIPETTES, getTipTypeFromTipRackDefinition, } from '@opentrons/shared-data' +import { ANALYTICS_QUICK_TRANSFER_SETTING_SAVED } from '/app/redux/analytics' -import { getTopPortalEl } from '../../../App/portal' +import { getTopPortalEl } from '/app/App/portal' import { ChildNavigation } from '../../ChildNavigation' import { NumericalKeyboard } from '/app/atoms/SoftwareKeyboard' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' + import { ACTIONS } from '../constants' import type { SupportedTip } from '@opentrons/shared-data' @@ -38,6 +41,7 @@ interface FlowRateEntryProps { export function FlowRateEntry(props: FlowRateEntryProps): JSX.Element { const { onBack, state, dispatch, kind } = props const { i18n, t } = useTranslation(['quick_transfer', 'shared']) + const { trackEventWithRobotSerial } = useTrackEventWithRobotSerial() const keyboardRef = React.useRef(null) const [flowRate, setFlowRate] = React.useState( @@ -88,6 +92,12 @@ export function FlowRateEntry(props: FlowRateEntryProps): JSX.Element { type: flowRateAction, rate: flowRate, }) + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_SETTING_SAVED, + properties: { + setting: `FlowRate_${kind}`, + }, + }) } onBack() } @@ -142,7 +152,7 @@ export function FlowRateEntry(props: FlowRateEntryProps): JSX.Element { > { setFlowRate(Number(e)) }} diff --git a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/Mix.tsx b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/Mix.tsx index c4413cdc04d..6755ba70f9a 100644 --- a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/Mix.tsx +++ b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/Mix.tsx @@ -13,8 +13,10 @@ import { SPACING, } from '@opentrons/components' -import { getTopPortalEl } from '../../../App/portal' +import { ANALYTICS_QUICK_TRANSFER_SETTING_SAVED } from '/app/redux/analytics' +import { getTopPortalEl } from '/app/App/portal' import { ChildNavigation } from '../../ChildNavigation' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' import { ACTIONS } from '../constants' import type { @@ -35,6 +37,7 @@ interface MixProps { export function Mix(props: MixProps): JSX.Element { const { kind, onBack, state, dispatch } = props const { t } = useTranslation('quick_transfer') + const { trackEventWithRobotSerial } = useTrackEventWithRobotSerial() const keyboardRef = React.useRef(null) const [mixIsEnabled, setMixIsEnabled] = React.useState( @@ -87,6 +90,12 @@ export function Mix(props: MixProps): JSX.Element { type: mixAction, mixSettings: undefined, }) + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_SETTING_SAVED, + properties: { + setting: `Mix_${kind}`, + }, + }) onBack() } else { setCurrentStep(2) @@ -99,6 +108,12 @@ export function Mix(props: MixProps): JSX.Element { type: mixAction, mixSettings: { mixVolume, repititions: mixReps }, }) + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_SETTING_SAVED, + properties: { + setting: `Mix_${kind}`, + }, + }) } onBack() } @@ -204,7 +219,7 @@ export function Mix(props: MixProps): JSX.Element { > { setMixVolume(Number(e)) }} diff --git a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/PipettePath.tsx b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/PipettePath.tsx index 86ef6cddf1a..ae90be4406e 100644 --- a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/PipettePath.tsx +++ b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/PipettePath.tsx @@ -15,8 +15,10 @@ import { } from '@opentrons/components' import { useNotifyDeckConfigurationQuery } from '/app/resources/deck_configuration' -import { getTopPortalEl } from '../../../App/portal' +import { ANALYTICS_QUICK_TRANSFER_SETTING_SAVED } from '/app/redux/analytics' +import { getTopPortalEl } from '/app/App/portal' import { ChildNavigation } from '../../ChildNavigation' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' import { useBlowOutLocationOptions } from './BlowOut' import { ACTIONS } from '../constants' @@ -39,6 +41,7 @@ interface PipettePathProps { export function PipettePath(props: PipettePathProps): JSX.Element { const { onBack, state, dispatch } = props const { t } = useTranslation('quick_transfer') + const { trackEventWithRobotSerial } = useTrackEventWithRobotSerial() const keyboardRef = React.useRef(null) const deckConfig = useNotifyDeckConfigurationQuery().data ?? [] @@ -48,9 +51,9 @@ export function PipettePath(props: PipettePathProps): JSX.Element { BlowOutLocation | undefined >(state.blowOut) - const [disposalVolume, setDisposalVolume] = React.useState( - state.volume - ) + const [disposalVolume, setDisposalVolume] = React.useState< + number | undefined + >(state?.disposalVolume) const maxPipetteVolume = Object.values(state.pipette.liquids)[0].maxVolume const tipVolume = Object.values(state.tipRack.wells)[0].totalLiquidVolume @@ -98,6 +101,12 @@ export function PipettePath(props: PipettePathProps): JSX.Element { type: ACTIONS.SET_PIPETTE_PATH, path: selectedPath, }) + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_SETTING_SAVED, + properties: { + setting: `PipettePath`, + }, + }) onBack() } else { setCurrentStep(2) @@ -111,6 +120,12 @@ export function PipettePath(props: PipettePathProps): JSX.Element { disposalVolume, blowOutLocation, }) + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_SETTING_SAVED, + properties: { + setting: `PipettePath`, + }, + }) onBack() } } @@ -124,7 +139,7 @@ export function PipettePath(props: PipettePathProps): JSX.Element { const volumeRange = { min: 1, max: maxDisposalCapacity } const volumeError = - disposalVolume !== null && + disposalVolume != null && (disposalVolume < volumeRange.min || disposalVolume > volumeRange.max) ? t(`value_out_of_range`, { min: volumeRange.min, @@ -203,7 +218,7 @@ export function PipettePath(props: PipettePathProps): JSX.Element { > { setDisposalVolume(Number(e)) }} diff --git a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/TipPosition.tsx b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/TipPosition.tsx index 3d831200bca..89e77cabc04 100644 --- a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/TipPosition.tsx +++ b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/TipPosition.tsx @@ -11,11 +11,13 @@ import { SPACING, } from '@opentrons/components' -import { getTopPortalEl } from '../../../App/portal' +import { ANALYTICS_QUICK_TRANSFER_SETTING_SAVED } from '/app/redux/analytics' +import { getTopPortalEl } from '/app/App/portal' import { ChildNavigation } from '../../ChildNavigation' import { NumericalKeyboard } from '/app/atoms/SoftwareKeyboard' import { ACTIONS } from '../constants' import { createPortal } from 'react-dom' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' import type { QuickTransferSummaryState, @@ -33,6 +35,7 @@ interface TipPositionEntryProps { export function TipPositionEntry(props: TipPositionEntryProps): JSX.Element { const { onBack, state, dispatch, kind } = props const { i18n, t } = useTranslation(['quick_transfer', 'shared']) + const { trackEventWithRobotSerial } = useTrackEventWithRobotSerial() const keyboardRef = React.useRef(null) const [tipPosition, setTipPosition] = React.useState( @@ -74,6 +77,12 @@ export function TipPositionEntry(props: TipPositionEntryProps): JSX.Element { type: tipPositionAction, position: tipPosition, }) + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_SETTING_SAVED, + properties: { + setting: `TipPosition_${kind}`, + }, + }) } onBack() } @@ -133,7 +142,7 @@ export function TipPositionEntry(props: TipPositionEntryProps): JSX.Element { > { setTipPosition(Number(e)) }} diff --git a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/TouchTip.tsx b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/TouchTip.tsx index 1bc81c09c97..7391f6da230 100644 --- a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/TouchTip.tsx +++ b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/TouchTip.tsx @@ -13,10 +13,12 @@ import { SPACING, } from '@opentrons/components' -import { getTopPortalEl } from '../../../App/portal' +import { ANALYTICS_QUICK_TRANSFER_SETTING_SAVED } from '/app/redux/analytics' +import { getTopPortalEl } from '/app/App/portal' import { ChildNavigation } from '../../ChildNavigation' import { ACTIONS } from '../constants' import { i18n } from '/app/i18n' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' import { NumericalKeyboard } from '/app/atoms/SoftwareKeyboard' import type { @@ -35,6 +37,7 @@ interface TouchTipProps { export function TouchTip(props: TouchTipProps): JSX.Element { const { kind, onBack, state, dispatch } = props const { t } = useTranslation('quick_transfer') + const { trackEventWithRobotSerial } = useTrackEventWithRobotSerial() const keyboardRef = React.useRef(null) const [touchTipIsEnabled, setTouchTipIsEnabled] = React.useState( @@ -79,12 +82,24 @@ export function TouchTip(props: TouchTipProps): JSX.Element { if (currentStep === 1) { if (!touchTipIsEnabled) { dispatch({ type: touchTipAction, position: undefined }) + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_SETTING_SAVED, + properties: { + setting: `TouchTip_${kind}`, + }, + }) onBack() } else { setCurrentStep(2) } } else if (currentStep === 2) { dispatch({ type: touchTipAction, position: position ?? undefined }) + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_SETTING_SAVED, + properties: { + setting: `TouchTip_${kind}`, + }, + }) onBack() } } @@ -196,7 +211,7 @@ export function TouchTip(props: TouchTipProps): JSX.Element { > { setPosition(Number(e)) }} diff --git a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/index.tsx b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/index.tsx index de45a16fff1..2911b7975d1 100644 --- a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/index.tsx +++ b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/index.tsx @@ -1,5 +1,6 @@ import * as React from 'react' import { useTranslation } from 'react-i18next' + import { ALIGN_CENTER, COLORS, @@ -19,6 +20,9 @@ import { TRASH_BIN_ADAPTER_FIXTURE, WASTE_CHUTE_FIXTURES, } from '@opentrons/shared-data' + +import { ANALYTICS_QUICK_TRANSFER_ADVANCED_SETTINGS_TAB } from '/app/redux/analytics' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' import { ACTIONS } from '../constants' import { useToaster } from '/app/organisms/ToasterOven' import { FlowRateEntry } from './FlowRate' @@ -47,8 +51,16 @@ export function QuickTransferAdvancedSettings( const [selectedSetting, setSelectedSetting] = React.useState( null ) + const { trackEventWithRobotSerial } = useTrackEventWithRobotSerial() const { makeSnackbar } = useToaster() + React.useEffect(() => { + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_ADVANCED_SETTINGS_TAB, + properties: {}, + }) + }, []) + function getBlowoutValueCopy(): string | undefined { if (state.blowOut === 'dest_well') { return t('blow_out_into_destination_well') diff --git a/app/src/organisms/QuickTransferFlow/SaveOrRunModal.tsx b/app/src/organisms/QuickTransferFlow/SaveOrRunModal.tsx index 46f4c63d7ee..c9e57469ac0 100644 --- a/app/src/organisms/QuickTransferFlow/SaveOrRunModal.tsx +++ b/app/src/organisms/QuickTransferFlow/SaveOrRunModal.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { SPACING, @@ -19,7 +19,8 @@ interface SaveOrRunModalProps { export const SaveOrRunModal = (props: SaveOrRunModalProps): JSX.Element => { const { t } = useTranslation('quick_transfer') - const [showNameTransfer, setShowNameTransfer] = React.useState(false) + const [showNameTransfer, setShowNameTransfer] = useState(false) + const [isLoading, setIsLoading] = useState(false) return showNameTransfer ? ( @@ -43,6 +44,7 @@ export const SaveOrRunModal = (props: SaveOrRunModalProps): JSX.Element => { { setShowNameTransfer(true) }} @@ -51,7 +53,11 @@ export const SaveOrRunModal = (props: SaveOrRunModalProps): JSX.Element => { { + setIsLoading(true) + props.onRun() + }} /> diff --git a/app/src/organisms/QuickTransferFlow/SelectDestWells.tsx b/app/src/organisms/QuickTransferFlow/SelectDestWells.tsx index d05f8993bc3..de18fafbc73 100644 --- a/app/src/organisms/QuickTransferFlow/SelectDestWells.tsx +++ b/app/src/organisms/QuickTransferFlow/SelectDestWells.tsx @@ -11,7 +11,9 @@ import { JUSTIFY_CENTER, } from '@opentrons/components' import { getAllDefinitions } from '@opentrons/shared-data' +import { ANALYTICS_QUICK_TRANSFER_WELL_SELECTION_DURATION } from '../../redux/analytics' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' import { getTopPortalEl } from '../../App/portal' import { OddModal } from '/app/molecules/OddModal' import { ChildNavigation } from '/app/organisms/ChildNavigation' @@ -39,6 +41,7 @@ interface SelectDestWellsProps { export function SelectDestWells(props: SelectDestWellsProps): JSX.Element { const { onNext, onBack, state, dispatch } = props const { i18n, t } = useTranslation(['quick_transfer', 'shared']) + const { trackEventWithRobotSerial } = useTrackEventWithRobotSerial() const { makeToast } = useToaster() @@ -72,6 +75,21 @@ export function SelectDestWells(props: SelectDestWellsProps): JSX.Element { selectionUnits = t('grids') } + let labwareDefinition = + state.destination === 'source' ? state.source : state.destination + if (labwareDefinition?.parameters.format === '96Standard') { + const allDefinitions = getAllDefinitions() + if (Object.values(labwareDefinition.wells)[0].shape === 'circular') { + labwareDefinition = allDefinitions[CIRCULAR_WELL_96_PLATE_DEFINITION_URI] + } else { + labwareDefinition = + allDefinitions[RECTANGULAR_WELL_96_PLATE_DEFINITION_URI] + } + } + const is384WellPlate = labwareDefinition?.parameters.format === '384Standard' + + const [analyticsStartTime] = React.useState(new Date()) + const handleClickNext = (): void => { if ( selectedWellCount === 1 || @@ -82,6 +100,14 @@ export function SelectDestWells(props: SelectDestWellsProps): JSX.Element { type: 'SET_DEST_WELLS', wells: Object.keys(selectedWells), }) + const duration = new Date().getTime() - analyticsStartTime.getTime() + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_WELL_SELECTION_DURATION, + properties: { + is384WellPlate, + duration: `${duration / 1000} seconds`, + }, + }) onNext() } else { setIsNumberWellsSelectedError(true) @@ -113,17 +139,7 @@ export function SelectDestWells(props: SelectDestWellsProps): JSX.Element { e.currentTarget.blur?.() }, } - let labwareDefinition = - state.destination === 'source' ? state.source : state.destination - if (labwareDefinition?.parameters.format === '96Standard') { - const allDefinitions = getAllDefinitions() - if (Object.values(labwareDefinition.wells)[0].shape === 'circular') { - labwareDefinition = allDefinitions[CIRCULAR_WELL_96_PLATE_DEFINITION_URI] - } else { - labwareDefinition = - allDefinitions[RECTANGULAR_WELL_96_PLATE_DEFINITION_URI] - } - } + return ( <> {createPortal( @@ -159,13 +175,7 @@ export function SelectDestWells(props: SelectDestWellsProps): JSX.Element { width="100%" > {labwareDefinition != null ? ( - + { diff --git a/app/src/organisms/QuickTransferFlow/SelectPipette.tsx b/app/src/organisms/QuickTransferFlow/SelectPipette.tsx index 18831b06dd3..9e5d0511b0f 100644 --- a/app/src/organisms/QuickTransferFlow/SelectPipette.tsx +++ b/app/src/organisms/QuickTransferFlow/SelectPipette.tsx @@ -10,7 +10,7 @@ import { } from '@opentrons/components' import { useInstrumentsQuery } from '@opentrons/react-api-client' import { RIGHT, LEFT } from '@opentrons/shared-data' -import { usePipetteSpecsV2 } from '/app/resources/instruments/hooks' +import { usePipetteSpecsV2 } from '/app/local-resources/instruments' import { ChildNavigation } from '../ChildNavigation' import type { PipetteData, Mount } from '@opentrons/api-client' diff --git a/app/src/organisms/QuickTransferFlow/SelectSourceWells.tsx b/app/src/organisms/QuickTransferFlow/SelectSourceWells.tsx index 9e23f101c83..56754e37ad2 100644 --- a/app/src/organisms/QuickTransferFlow/SelectSourceWells.tsx +++ b/app/src/organisms/QuickTransferFlow/SelectSourceWells.tsx @@ -11,6 +11,8 @@ import { getAllDefinitions } from '@opentrons/shared-data' import { ChildNavigation } from '/app/organisms/ChildNavigation' import { WellSelection } from '/app/organisms/WellSelection' +import { ANALYTICS_QUICK_TRANSFER_WELL_SELECTION_DURATION } from '/app/redux/analytics' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' import type { SmallButton } from '/app/atoms/buttons' import type { @@ -33,6 +35,7 @@ export const RECTANGULAR_WELL_96_PLATE_DEFINITION_URI = export function SelectSourceWells(props: SelectSourceWellsProps): JSX.Element { const { onNext, onBack, state, dispatch } = props const { i18n, t } = useTranslation(['quick_transfer', 'shared']) + const { trackEventWithRobotSerial } = useTrackEventWithRobotSerial() const sourceWells = state.sourceWells ?? [] const sourceWellGroup = sourceWells.reduce((acc, well) => { @@ -40,12 +43,22 @@ export function SelectSourceWells(props: SelectSourceWellsProps): JSX.Element { }, {}) const [selectedWells, setSelectedWells] = React.useState(sourceWellGroup) + const [startingTimeStamp] = React.useState(new Date()) + const is384WellPlate = state.source?.parameters.format === '384Standard' const handleClickNext = (): void => { dispatch({ type: 'SET_SOURCE_WELLS', wells: Object.keys(selectedWells), }) + const duration = new Date().getTime() - startingTimeStamp?.getTime() + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_WELL_SELECTION_DURATION, + properties: { + is384WellPlate, + duration: `${duration / 1000} seconds`, + }, + }) onNext() } @@ -91,11 +104,7 @@ export function SelectSourceWells(props: SelectSourceWellsProps): JSX.Element { width="100%" > {state.source != null && displayLabwareDefinition != null ? ( - + { diff --git a/app/src/organisms/QuickTransferFlow/SummaryAndSettings.tsx b/app/src/organisms/QuickTransferFlow/SummaryAndSettings.tsx index cfc7d96e21d..b8a345099fb 100644 --- a/app/src/organisms/QuickTransferFlow/SummaryAndSettings.tsx +++ b/app/src/organisms/QuickTransferFlow/SummaryAndSettings.tsx @@ -17,8 +17,14 @@ import { useCreateRunMutation, useHost, } from '@opentrons/react-api-client' -import { useNotifyDeckConfigurationQuery } from '/app/resources/deck_configuration' +import { useNotifyDeckConfigurationQuery } from '/app/resources/deck_configuration' +import { + ANALYTICS_QUICK_TRANSFER_TIME_TO_CREATE, + ANALYTICS_QUICK_TRANSFER_SAVE_FOR_LATER, + ANALYTICS_QUICK_TRANSFER_RUN_NOW, +} from '/app/redux/analytics' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' import { ChildNavigation } from '../ChildNavigation' import { Overview } from './Overview' import { TipManagement } from './TipManagement' @@ -33,13 +39,15 @@ import type { QuickTransferWizardState } from './types' interface SummaryAndSettingsProps { exitButtonProps: React.ComponentProps state: QuickTransferWizardState + analyticsStartTime: Date } export function SummaryAndSettings( props: SummaryAndSettingsProps ): JSX.Element | null { - const { exitButtonProps, state: wizardFlowState } = props + const { exitButtonProps, state: wizardFlowState, analyticsStartTime } = props const navigate = useNavigate() + const { trackEventWithRobotSerial } = useTrackEventWithRobotSerial() const queryClient = useQueryClient() const host = useHost() const { t } = useTranslation(['quick_transfer', 'shared']) @@ -82,6 +90,17 @@ export function SummaryAndSettings( host ) + const handleClickCreateTransfer = (): void => { + setShowSaveOrRunModal(true) + const duration = new Date().getTime() - analyticsStartTime.getTime() + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_TIME_TO_CREATE, + properties: { + duration: `${duration / 1000} seconds`, + }, + }) + } + const handleClickSave = (protocolName: string): void => { const protocolFile = createQuickTransferFile( state, @@ -94,6 +113,12 @@ export function SummaryAndSettings( }).then(() => { navigate('/quick-transfer') }) + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_SAVE_FOR_LATER, + properties: { + name: protocolName, + }, + }) } const handleClickRun = (): void => { @@ -106,6 +131,10 @@ export function SummaryAndSettings( protocolId: data.data.id, }) }) + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_RUN_NOW, + properties: {}, + }) } return showSaveOrRunModal ? ( @@ -115,9 +144,7 @@ export function SummaryAndSettings( { - setShowSaveOrRunModal(true) - }} + onClickButton={handleClickCreateTransfer} secondaryButtonProps={exitButtonProps} /> ( null ) + React.useEffect(() => { + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_TIP_MANAGEMENT_TAB, + properties: {}, + }) + }, []) + const displayItems = [ { option: t('change_tip'), diff --git a/app/src/organisms/QuickTransferFlow/__tests__/ConfirmExitModal.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/ConfirmExitModal.test.tsx index a35a2b3eb5f..4b50c31ca29 100644 --- a/app/src/organisms/QuickTransferFlow/__tests__/ConfirmExitModal.test.tsx +++ b/app/src/organisms/QuickTransferFlow/__tests__/ConfirmExitModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/QuickTransferFlow/__tests__/CreateNewTransfer.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/CreateNewTransfer.test.tsx index 0b72abd3627..178086ae401 100644 --- a/app/src/organisms/QuickTransferFlow/__tests__/CreateNewTransfer.test.tsx +++ b/app/src/organisms/QuickTransferFlow/__tests__/CreateNewTransfer.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' import { DeckConfigurator } from '@opentrons/components' diff --git a/app/src/organisms/QuickTransferFlow/__tests__/NameQuickTransfer.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/NameQuickTransfer.test.tsx index 58339b4abb8..363c89cdc82 100644 --- a/app/src/organisms/QuickTransferFlow/__tests__/NameQuickTransfer.test.tsx +++ b/app/src/organisms/QuickTransferFlow/__tests__/NameQuickTransfer.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' @@ -52,6 +52,7 @@ describe('NameQuickTransfer', () => { expect(saveBtn).toBeEnabled() fireEvent.click(saveBtn) expect(props.onSave).toHaveBeenCalled() + expect(saveBtn).toBeDisabled() }) it('disables save if you enter more than 60 characters', () => { diff --git a/app/src/organisms/QuickTransferFlow/__tests__/Overview.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/Overview.test.tsx index a60804414f4..242b0f58a92 100644 --- a/app/src/organisms/QuickTransferFlow/__tests__/Overview.test.tsx +++ b/app/src/organisms/QuickTransferFlow/__tests__/Overview.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, afterEach, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/AirGap.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/AirGap.test.tsx new file mode 100644 index 00000000000..b3e7ef55e31 --- /dev/null +++ b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/AirGap.test.tsx @@ -0,0 +1,276 @@ +import type * as React from 'react' +import { fireEvent, screen } from '@testing-library/react' +import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' + +import { InputField } from '@opentrons/components' + +import { renderWithProviders } from '../../../../__testing-utils__' +import { i18n } from '../../../../i18n' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' +import { AirGap } from '../../QuickTransferAdvancedSettings/AirGap' +import type { QuickTransferSummaryState } from '../../types' + +vi.mock('/app/redux-resources/analytics') +vi.mock('../utils') + +vi.mock('@opentrons/components', async importOriginal => { + const actualComponents = await importOriginal() + return { + ...actualComponents, + InputField: vi.fn(), + } +}) + +const render = (props: React.ComponentProps) => { + return renderWithProviders(, { + i18nInstance: i18n, + }) +} +let mockTrackEventWithRobotSerial: any + +describe('AirGap', () => { + let props: React.ComponentProps + + beforeEach(() => { + props = { + onBack: vi.fn(), + kind: 'aspirate', + state: { + mount: 'left', + pipette: { + channels: 1, + liquids: [ + { + maxVolume: 1000, + minVolume: 5, + }, + ] as any, + } as any, + tipRack: { + wells: { + A1: { + totalLiquidVolume: 200, + }, + } as any, + } as any, + sourceWells: ['A1'], + destinationWells: ['A1'], + transferType: 'transfer', + volume: 20, + path: 'single', + } as QuickTransferSummaryState, + dispatch: vi.fn(), + } + mockTrackEventWithRobotSerial = vi.fn( + () => new Promise(resolve => resolve({})) + ) + vi.mocked(useTrackEventWithRobotSerial).mockReturnValue({ + trackEventWithRobotSerial: mockTrackEventWithRobotSerial, + }) + }) + afterEach(() => { + vi.resetAllMocks() + }) + + it('renders the first air gap screen, continue, and back buttons', () => { + render(props) + screen.getByText('Air gap before aspirating') + screen.getByTestId('ChildNavigation_Primary_Button') + screen.getByText('Enabled') + screen.getByText('Disabled') + const exitBtn = screen.getByTestId('ChildNavigation_Back_Button') + fireEvent.click(exitBtn) + expect(props.onBack).toHaveBeenCalled() + }) + + it('renders save button if you select enabled, then moves to second screen', () => { + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Air gap volume (µL)', + error: null, + readOnly: true, + type: 'number', + value: null, + }, + {} + ) + }) + + it('calls dispatch button if you select disabled and save', () => { + render(props) + const disabledBtn = screen.getByText('Disabled') + fireEvent.click(disabledBtn) + const saveBtn = screen.getByText('Save') + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() + }) + + it('has correct range for aspirate with a single pipette path', () => { + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const numButton = screen.getByText('0') + fireEvent.click(numButton) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Air gap volume (µL)', + error: 'Value must be between 1-180', + readOnly: true, + type: 'number', + value: 0, + }, + {} + ) + const saveBtn = screen.getByTestId('ChildNavigation_Primary_Button') + expect(saveBtn).toBeDisabled() + }) + + it('has correct range for aspirate with a multiAspirate pipette path', () => { + props = { + ...props, + state: { + ...props.state, + path: 'multiAspirate', + }, + } + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const numButton = screen.getByText('0') + fireEvent.click(numButton) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Air gap volume (µL)', + error: 'Value must be between 1-80', + readOnly: true, + type: 'number', + value: 0, + }, + {} + ) + }) + + it('has correct range for aspirate with a multiDispense pipette path', () => { + props = { + ...props, + state: { + ...props.state, + path: 'multiDispense', + }, + } + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const numButton = screen.getByText('0') + fireEvent.click(numButton) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Air gap volume (µL)', + error: 'Value must be between 1-140', + readOnly: true, + type: 'number', + value: 0, + }, + {} + ) + }) + + it('has correct range for and text for a dispense', () => { + props = { + ...props, + kind: 'dispense', + } + render(props) + screen.getByText('Air gap before dispensing') + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const numButton = screen.getByText('0') + fireEvent.click(numButton) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Air gap volume (µL)', + error: 'Value must be between 1-200', + readOnly: true, + type: 'number', + value: 0, + }, + {} + ) + }) + + it('calls dispatch when an in range value is entered and saved', () => { + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const numButton = screen.getByText('1') + fireEvent.click(numButton) + const saveBtn = screen.getByText('Save') + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() + }) + + it('persists existing values if they are in state for aspirate', () => { + props = { + ...props, + state: { + ...props.state, + airGapAspirate: 4, + }, + } + render(props) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Air gap volume (µL)', + error: null, + readOnly: true, + type: 'number', + value: 4, + }, + {} + ) + }) + + it('persists existing values if they are in state for dispense', () => { + props = { + ...props, + kind: 'dispense', + state: { + ...props.state, + airGapAspirate: 4, + airGapDispense: 16, + }, + } + render(props) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Air gap volume (µL)', + error: null, + readOnly: true, + type: 'number', + value: 16, + }, + {} + ) + }) +}) diff --git a/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/BlowOut.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/BlowOut.test.tsx new file mode 100644 index 00000000000..48d32beeb31 --- /dev/null +++ b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/BlowOut.test.tsx @@ -0,0 +1,171 @@ +import type * as React from 'react' +import { fireEvent, screen } from '@testing-library/react' +import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' + +import { renderWithProviders } from '/app/__testing-utils__' +import { i18n } from '../../../../i18n' +import { useNotifyDeckConfigurationQuery } from '../../../../resources/deck_configuration' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' +import { BlowOut } from '../../QuickTransferAdvancedSettings/BlowOut' +import type { QuickTransferSummaryState } from '../../types' + +vi.mock('../../../../resources/deck_configuration') +vi.mock('/app/redux-resources/analytics') +vi.mock('../utils') + +const render = (props: React.ComponentProps) => { + return renderWithProviders(, { + i18nInstance: i18n, + }) +} +let mockTrackEventWithRobotSerial: any + +describe('BlowOut', () => { + let props: React.ComponentProps + + beforeEach(() => { + props = { + kind: 'aspirate', + onBack: vi.fn(), + state: { + mount: 'left', + pipette: { + channels: 1, + liquids: [ + { + maxVolume: 1000, + minVolume: 5, + }, + ] as any, + } as any, + tipRack: { + wells: { + A1: { + totalLiquidVolume: 200, + }, + } as any, + } as any, + sourceWells: ['A1'], + destinationWells: ['A1'], + transferType: 'transfer', + volume: 20, + path: 'single', + } as QuickTransferSummaryState, + dispatch: vi.fn(), + } + mockTrackEventWithRobotSerial = vi.fn( + () => new Promise(resolve => resolve({})) + ) + vi.mocked(useTrackEventWithRobotSerial).mockReturnValue({ + trackEventWithRobotSerial: mockTrackEventWithRobotSerial, + }) + vi.mocked(useNotifyDeckConfigurationQuery).mockReturnValue({ + data: [ + { + cutoutId: 'cutoutC3', + cutoutFixtureId: 'wasteChuteRightAdapterCovered', + }, + { + cutoutId: 'cutoutA3', + cutoutFixtureId: 'trashBinAdapter', + }, + ], + } as any) + }) + afterEach(() => { + vi.resetAllMocks() + }) + + it('renders the first blow out screen, continue, and back buttons', () => { + render(props) + screen.getByText('Blowout after dispensing') + screen.getByTestId('ChildNavigation_Primary_Button') + screen.getByText('Enabled') + screen.getByText('Disabled') + const exitBtn = screen.getByTestId('ChildNavigation_Back_Button') + fireEvent.click(exitBtn) + expect(props.onBack).toHaveBeenCalled() + }) + + it('calls dispatch button if you select disabled and save', () => { + render(props) + const disabledBtn = screen.getByText('Disabled') + fireEvent.click(disabledBtn) + const saveBtn = screen.getByText('Save') + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + }) + + it('second screen renders both source and destination wells and deck config trash options for transfer', () => { + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + screen.getByText('Source well') + screen.getByText('Destination well') + screen.getByText('Trash bin in A3') + screen.getByText('Waste chute in C3') + }) + + it('second screen renders trash bin in A3 if deck config is empty', () => { + vi.mocked(useNotifyDeckConfigurationQuery).mockReturnValue({ + data: [] as any, + } as any) + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + screen.getByText('Trash bin in A3') + }) + + it('second screen renders source well but not dest well for distribute', () => { + props = { + ...props, + state: { + ...props.state, + transferType: 'distribute', + }, + } + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + screen.getByText('Source well') + expect(screen.queryByText('Destination well')).not.toBeInTheDocument() + }) + + it('second screen renders dest well but not source well for consolidate', () => { + props = { + ...props, + state: { + ...props.state, + transferType: 'consolidate', + }, + } + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + screen.getByText('Destination well') + expect(screen.queryByText('Source well')).not.toBeInTheDocument() + }) + + it('enables save button when you make a destination selection and calls dispatch when saved', () => { + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const saveBtn = screen.getByTestId('ChildNavigation_Primary_Button') + expect(saveBtn).toBeDisabled() + const destBtn = screen.getByText('Destination well') + fireEvent.click(destBtn) + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() + }) +}) diff --git a/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/Delay.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/Delay.test.tsx new file mode 100644 index 00000000000..58f1876adab --- /dev/null +++ b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/Delay.test.tsx @@ -0,0 +1,293 @@ +import type * as React from 'react' +import { fireEvent, screen } from '@testing-library/react' +import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' + +import { InputField } from '@opentrons/components' + +import { renderWithProviders } from '/app/__testing-utils__' +import { i18n } from '../../../../i18n' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' +import { Delay } from '../../QuickTransferAdvancedSettings/Delay' +import type { QuickTransferSummaryState } from '../../types' + +vi.mock('/app/redux-resources/analytics') +vi.mock('../utils') + +vi.mock('@opentrons/components', async importOriginal => { + const actualComponents = await importOriginal() + return { + ...actualComponents, + InputField: vi.fn(), + } +}) + +const render = (props: React.ComponentProps) => { + return renderWithProviders(, { + i18nInstance: i18n, + }) +} +let mockTrackEventWithRobotSerial: any + +describe('Delay', () => { + let props: React.ComponentProps + + beforeEach(() => { + props = { + onBack: vi.fn(), + kind: 'aspirate', + state: { + mount: 'left', + pipette: { + channels: 1, + liquids: [ + { + maxVolume: 1000, + minVolume: 5, + }, + ] as any, + } as any, + source: { + wells: { + A1: { + totalLiquidVolume: 200, + depth: 50, + }, + } as any, + } as any, + destination: { + wells: { + A1: { + totalLiquidVolume: 200, + depth: 200, + }, + } as any, + } as any, + sourceWells: ['A1'], + destinationWells: ['A1'], + transferType: 'transfer', + volume: 20, + path: 'single', + } as QuickTransferSummaryState, + dispatch: vi.fn(), + } + mockTrackEventWithRobotSerial = vi.fn( + () => new Promise(resolve => resolve({})) + ) + vi.mocked(useTrackEventWithRobotSerial).mockReturnValue({ + trackEventWithRobotSerial: mockTrackEventWithRobotSerial, + }) + }) + afterEach(() => { + vi.resetAllMocks() + }) + + it('renders the first delay screen, continue, and back buttons', () => { + render(props) + screen.getByText('Delay before aspirating') + screen.getByTestId('ChildNavigation_Primary_Button') + screen.getByText('Enabled') + screen.getByText('Disabled') + const exitBtn = screen.getByTestId('ChildNavigation_Back_Button') + fireEvent.click(exitBtn) + expect(props.onBack).toHaveBeenCalled() + }) + + it('renders save button if you select enabled, then moves to second screen', () => { + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Delay duration (seconds)', + error: null, + readOnly: true, + type: 'number', + value: null, + }, + {} + ) + }) + + it('calls dispatch button if you select disabled and save', () => { + render(props) + const disabledBtn = screen.getByText('Disabled') + fireEvent.click(disabledBtn) + const saveBtn = screen.getByText('Save') + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() + }) + + it('has correct delay duration range', () => { + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const oneButton = screen.getByText('0') + fireEvent.click(oneButton) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Delay duration (seconds)', + error: 'Value must be between 1-9999999999', + readOnly: true, + type: 'number', + value: 0, + }, + {} + ) + const nextBtn = screen.getByTestId('ChildNavigation_Primary_Button') + expect(nextBtn).toBeDisabled() + }) + + it('has correct range for delay height for aspirate', () => { + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const oneButton = screen.getByText('1') + fireEvent.click(oneButton) + const nextBtn = screen.getByTestId('ChildNavigation_Primary_Button') + fireEvent.click(nextBtn) + const zeroButton = screen.getByText('0') + fireEvent.click(zeroButton) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Delay position from bottom of well (mm)', + error: 'Value must be between 1-100', + readOnly: true, + type: 'number', + value: 0, + }, + {} + ) + const saveBtn = screen.getByTestId('ChildNavigation_Primary_Button') + expect(saveBtn).toBeDisabled() + }) + + it('has correct range for delay height for dispense', () => { + props = { + ...props, + kind: 'dispense', + } + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const oneButton = screen.getByText('1') + fireEvent.click(oneButton) + const nextBtn = screen.getByTestId('ChildNavigation_Primary_Button') + fireEvent.click(nextBtn) + const zeroButton = screen.getByText('0') + fireEvent.click(zeroButton) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Delay position from bottom of well (mm)', + error: 'Value must be between 1-400', + readOnly: true, + type: 'number', + value: 0, + }, + {} + ) + const saveBtn = screen.getByTestId('ChildNavigation_Primary_Button') + expect(saveBtn).toBeDisabled() + }) + + it('calls dispatch when an in range value is entered and saved', () => { + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const oneButton = screen.getByText('1') + fireEvent.click(oneButton) + const nextBtn = screen.getByTestId('ChildNavigation_Primary_Button') + fireEvent.click(nextBtn) + const twoButton = screen.getByText('2') + fireEvent.click(twoButton) + const saveBtn = screen.getByText('Save') + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() + }) + + it('persists previously set value saved in state for aspirate', () => { + props = { + ...props, + state: { + ...props.state, + delayAspirate: { + delayDuration: 15, + positionFromBottom: 55, + }, + }, + } + render(props) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Delay duration (seconds)', + error: null, + readOnly: true, + type: 'number', + value: 15, + }, + {} + ) + fireEvent.click(continueBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Delay position from bottom of well (mm)', + error: null, + readOnly: true, + type: 'number', + value: 55, + }, + {} + ) + }) + + it('persists previously set value saved in state for dispense', () => { + props = { + ...props, + kind: 'dispense', + state: { + ...props.state, + delayDispense: { + delayDuration: 20, + positionFromBottom: 84, + }, + }, + } + render(props) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Delay duration (seconds)', + error: null, + readOnly: true, + type: 'number', + value: 20, + }, + {} + ) + fireEvent.click(continueBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Delay position from bottom of well (mm)', + error: null, + readOnly: true, + type: 'number', + value: 84, + }, + {} + ) + }) +}) diff --git a/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/FlowRate.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/FlowRate.test.tsx new file mode 100644 index 00000000000..fc8b6960e1d --- /dev/null +++ b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/FlowRate.test.tsx @@ -0,0 +1,158 @@ +import type * as React from 'react' +import { fireEvent, screen } from '@testing-library/react' +import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' + +import { InputField } from '@opentrons/components' + +import { renderWithProviders } from '/app/__testing-utils__' +import { i18n } from '../../../../i18n' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' +import { FlowRateEntry } from '../../QuickTransferAdvancedSettings/FlowRate' +import type { QuickTransferSummaryState } from '../../types' + +vi.mock('/app/redux-resources/analytics') +vi.mock('../utils') + +vi.mock('@opentrons/components', async importOriginal => { + const actualComponents = await importOriginal() + return { + ...actualComponents, + InputField: vi.fn(), + } +}) + +const render = (props: React.ComponentProps) => { + return renderWithProviders(, { + i18nInstance: i18n, + }) +} +let mockTrackEventWithRobotSerial: any + +describe('FlowRate', () => { + let props: React.ComponentProps + + beforeEach(() => { + props = { + onBack: vi.fn(), + kind: 'aspirate', + state: { + mount: 'left', + pipette: { + model: 'p50', + channels: 1, + liquids: { + default: { + maxVolume: 1000, + minVolume: 5, + supportedTips: { + t50: { + uiMaxFlowRate: 92, + defaultAspirateFlowRate: { + default: 30, + }, + defaultDispenseFlowRate: { + default: 80, + }, + }, + }, + }, + } as any, + } as any, + tipRack: { + wells: { + A1: { + totalLiquidVolume: 50, + }, + } as any, + } as any, + sourceWells: ['A1'], + destinationWells: ['A1'], + transferType: 'transfer', + volume: 20, + path: 'single', + aspirateFlowRate: 35, + dispenseFlowRate: 62, + } as QuickTransferSummaryState, + dispatch: vi.fn(), + } + mockTrackEventWithRobotSerial = vi.fn( + () => new Promise(resolve => resolve({})) + ) + vi.mocked(useTrackEventWithRobotSerial).mockReturnValue({ + trackEventWithRobotSerial: mockTrackEventWithRobotSerial, + }) + }) + afterEach(() => { + vi.resetAllMocks() + }) + + it('renders the flow rate aspirate screen, continue, and back buttons', () => { + render(props) + screen.getByText('Aspirate flow rate') + screen.getByTestId('ChildNavigation_Primary_Button') + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Aspirate flow rate (µL/s)', + error: null, + readOnly: true, + type: 'number', + value: 35, + }, + {} + ) + const exitBtn = screen.getByTestId('ChildNavigation_Back_Button') + fireEvent.click(exitBtn) + expect(props.onBack).toHaveBeenCalled() + }) + + it('renders the flow rate dispense screen', () => { + props = { + ...props, + kind: 'dispense', + } + render(props) + screen.getByText('Dispense flow rate') + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Dispense flow rate (µL/s)', + error: null, + readOnly: true, + type: 'number', + value: 62, + }, + {} + ) + }) + + it('renders correct range if you enter incorrect value', () => { + render(props) + const deleteBtn = screen.getByText('del') + fireEvent.click(deleteBtn) + fireEvent.click(deleteBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Aspirate flow rate (µL/s)', + error: 'Value must be between 1-92', + readOnly: true, + type: 'number', + value: 0, + }, + {} + ) + const saveBtn = screen.getByTestId('ChildNavigation_Primary_Button') + expect(saveBtn).toBeDisabled() + }) + + it('calls dispatch when an in range value is entered and saved', () => { + render(props) + const deleteBtn = screen.getByText('del') + fireEvent.click(deleteBtn) + fireEvent.click(deleteBtn) + const numButton = screen.getByText('1') + fireEvent.click(numButton) + const saveBtn = screen.getByText('Save') + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() + }) +}) diff --git a/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/Mix.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/Mix.test.tsx new file mode 100644 index 00000000000..6f66cc911d6 --- /dev/null +++ b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/Mix.test.tsx @@ -0,0 +1,263 @@ +import type * as React from 'react' +import { fireEvent, screen } from '@testing-library/react' +import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' + +import { InputField } from '@opentrons/components' + +import { renderWithProviders } from '/app/__testing-utils__' +import { i18n } from '../../../../i18n' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' +import { Mix } from '../../QuickTransferAdvancedSettings/Mix' +import type { QuickTransferSummaryState } from '../../types' + +vi.mock('/app/redux-resources/analytics') +vi.mock('../utils') + +vi.mock('@opentrons/components', async importOriginal => { + const actualComponents = await importOriginal() + return { + ...actualComponents, + InputField: vi.fn(), + } +}) + +const render = (props: React.ComponentProps) => { + return renderWithProviders(, { + i18nInstance: i18n, + }) +} +let mockTrackEventWithRobotSerial: any + +describe('Mix', () => { + let props: React.ComponentProps + + beforeEach(() => { + props = { + onBack: vi.fn(), + kind: 'aspirate', + state: { + mount: 'left', + pipette: { + channels: 1, + liquids: [ + { + maxVolume: 1000, + minVolume: 5, + }, + ] as any, + } as any, + tipRack: { + wells: { + A1: { + totalLiquidVolume: 200, + }, + } as any, + } as any, + sourceWells: ['A1'], + destinationWells: ['A1'], + transferType: 'transfer', + volume: 20, + path: 'single', + } as QuickTransferSummaryState, + dispatch: vi.fn(), + } + mockTrackEventWithRobotSerial = vi.fn( + () => new Promise(resolve => resolve({})) + ) + vi.mocked(useTrackEventWithRobotSerial).mockReturnValue({ + trackEventWithRobotSerial: mockTrackEventWithRobotSerial, + }) + }) + afterEach(() => { + vi.resetAllMocks() + }) + + it('renders the first Mix screen, continue, and back buttons', () => { + render(props) + screen.getByText('Mix before aspirating') + screen.getByTestId('ChildNavigation_Primary_Button') + screen.getByText('Enabled') + screen.getByText('Disabled') + const exitBtn = screen.getByTestId('ChildNavigation_Back_Button') + fireEvent.click(exitBtn) + expect(props.onBack).toHaveBeenCalled() + }) + + it('renders the different copy for Mix on dispense', () => { + props = { + ...props, + kind: 'dispense', + } + render(props) + screen.getByText('Mix before dispensing') + }) + + it('renders save button if you select enabled, then moves to second screen', () => { + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Mix volume (µL)', + error: null, + readOnly: true, + type: 'number', + value: null, + }, + {} + ) + }) + + it('calls dispatch button if you select disabled and save', () => { + render(props) + const disabledBtn = screen.getByText('Disabled') + fireEvent.click(disabledBtn) + const saveBtn = screen.getByText('Save') + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() + }) + + it('has correct Mix volume range', () => { + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const oneButton = screen.getByText('0') + fireEvent.click(oneButton) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Mix volume (µL)', + error: 'Value must be between 1-200', + readOnly: true, + type: 'number', + value: 0, + }, + {} + ) + const nextBtn = screen.getByTestId('ChildNavigation_Primary_Button') + expect(nextBtn).toBeDisabled() + }) + + it('has correct range for Mix repitition range', () => { + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const oneButton = screen.getByText('1') + fireEvent.click(oneButton) + const nextBtn = screen.getByTestId('ChildNavigation_Primary_Button') + fireEvent.click(nextBtn) + const zeroButton = screen.getByText('0') + fireEvent.click(zeroButton) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Mix repetitions', + error: 'Value must be between 1-999', + readOnly: true, + type: 'number', + value: 0, + }, + {} + ) + const saveBtn = screen.getByTestId('ChildNavigation_Primary_Button') + expect(saveBtn).toBeDisabled() + }) + + it('calls dispatch when an in range value is entered and saved', () => { + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const oneButton = screen.getByText('1') + fireEvent.click(oneButton) + const nextBtn = screen.getByTestId('ChildNavigation_Primary_Button') + fireEvent.click(nextBtn) + const twoButton = screen.getByText('2') + fireEvent.click(twoButton) + const saveBtn = screen.getByText('Save') + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() + }) + + it('persists previously set value saved in state for aspirate', () => { + props = { + ...props, + state: { + ...props.state, + mixOnAspirate: { + mixVolume: 15, + repititions: 55, + }, + }, + } + render(props) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Mix volume (µL)', + error: null, + readOnly: true, + type: 'number', + value: 15, + }, + {} + ) + fireEvent.click(continueBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Mix repetitions', + error: null, + readOnly: true, + type: 'number', + value: 55, + }, + {} + ) + }) + + it('persists previously set value saved in state for dispense', () => { + props = { + ...props, + kind: 'dispense', + state: { + ...props.state, + mixOnDispense: { + mixVolume: 18, + repititions: 2, + }, + }, + } + render(props) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Mix volume (µL)', + error: null, + readOnly: true, + type: 'number', + value: 18, + }, + {} + ) + fireEvent.click(continueBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Mix repetitions', + error: null, + readOnly: true, + type: 'number', + value: 2, + }, + {} + ) + }) +}) diff --git a/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/PipettePath.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/PipettePath.test.tsx new file mode 100644 index 00000000000..985fc73cba3 --- /dev/null +++ b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/PipettePath.test.tsx @@ -0,0 +1,230 @@ +import type * as React from 'react' +import { fireEvent, screen } from '@testing-library/react' +import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' + +import { InputField } from '@opentrons/components' + +import { renderWithProviders } from '/app/__testing-utils__' +import { i18n } from '../../../../i18n' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' +import { PipettePath } from '../../QuickTransferAdvancedSettings/PipettePath' +import { useBlowOutLocationOptions } from '../../QuickTransferAdvancedSettings/BlowOut' +import type { QuickTransferSummaryState } from '../../types' + +vi.mock('/app/redux-resources/analytics') +vi.mock('../utils') +vi.mock('../../QuickTransferAdvancedSettings/BlowOut') + +vi.mock('@opentrons/components', async importOriginal => { + const actualComponents = await importOriginal() + return { + ...actualComponents, + InputField: vi.fn(), + } +}) + +const render = (props: React.ComponentProps) => { + return renderWithProviders(, { + i18nInstance: i18n, + }) +} +let mockTrackEventWithRobotSerial: any + +describe('PipettePath', () => { + let props: React.ComponentProps + + beforeEach(() => { + props = { + onBack: vi.fn(), + state: { + mount: 'left', + pipette: { + channels: 1, + liquids: [ + { + maxVolume: 1000, + minVolume: 5, + }, + ] as any, + } as any, + tipRack: { + wells: { + A1: { + totalLiquidVolume: 200, + }, + } as any, + } as any, + transferType: 'consolidate', + volume: 20, + path: 'multiAspirate', + } as QuickTransferSummaryState, + dispatch: vi.fn(), + } + mockTrackEventWithRobotSerial = vi.fn( + () => new Promise(resolve => resolve({})) + ) + vi.mocked(useTrackEventWithRobotSerial).mockReturnValue({ + trackEventWithRobotSerial: mockTrackEventWithRobotSerial, + }) + vi.mocked(useBlowOutLocationOptions).mockReturnValue([ + { + location: 'source_well', + description: 'Source well', + }, + ]) + }) + afterEach(() => { + vi.resetAllMocks() + }) + + it('renders the first pipette path screen, continue, back buttons', () => { + render(props) + screen.getByText('Pipette path') + screen.getByTestId('ChildNavigation_Primary_Button') + const exitBtn = screen.getByTestId('ChildNavigation_Back_Button') + fireEvent.click(exitBtn) + expect(props.onBack).toHaveBeenCalled() + }) + + it('renders multi aspirate and single options for consolidate if there is room in the tip', () => { + render(props) + screen.getByText('Single transfers') + screen.getByText('Multi-aspirate') + const saveBtn = screen.getByTestId('ChildNavigation_Primary_Button') + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() + }) + + it('renders single option only for consolidate if there is not room in the tip', () => { + props = { + ...props, + state: { + ...props.state, + volume: 101, + }, + } + render(props) + screen.getByText('Single transfers') + expect(screen.queryByText('Multi-aspirate')).not.toBeInTheDocument() + const saveBtn = screen.getByTestId('ChildNavigation_Primary_Button') + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() + }) + + it('renders multi dispense and single options for distribute if there is room in the tip', () => { + props = { + ...props, + state: { + ...props.state, + transferType: 'distribute', + }, + } + render(props) + screen.getByText('Single transfers') + screen.getByText('Multi-dispense') + const saveBtn = screen.getByTestId('ChildNavigation_Primary_Button') + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() + }) + + it('renders single option only for distribute if there is not room in the tip', () => { + props = { + ...props, + state: { + ...props.state, + transferType: 'distribute', + volume: 67, + }, + } + render(props) + screen.getByText('Single transfers') + expect(screen.queryByText('Multi-dispense')).not.toBeInTheDocument() + const saveBtn = screen.getByTestId('ChildNavigation_Primary_Button') + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() + }) + + it('renders next cta and disposal volume screen if you choose multi dispense', () => { + props = { + ...props, + state: { + ...props.state, + transferType: 'distribute', + disposalVolume: 20, + blowOut: 'source_well', + }, + } + render(props) + const multiDispenseBtn = screen.getByText('Multi-dispense') + fireEvent.click(multiDispenseBtn) + const continueBtn = screen.getByTestId('ChildNavigation_Primary_Button') + fireEvent.click(continueBtn) + + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Disposal volume (µL)', + error: null, + readOnly: true, + type: 'number', + value: 20, + }, + {} + ) + }) + + it('renders error on disposal volume screen if you select an out of range value', () => { + props = { + ...props, + state: { + ...props.state, + transferType: 'distribute', + path: 'multiDispense', + disposalVolume: 20, + blowOut: 'source_well', + }, + } + render(props) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const oneButton = screen.getByText('1') + fireEvent.click(oneButton) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Disposal volume (µL)', + error: 'Value must be between 1-160', + readOnly: true, + type: 'number', + value: 201, + }, + {} + ) + const nextBtn = screen.getByTestId('ChildNavigation_Primary_Button') + expect(nextBtn).toBeDisabled() + }) + + it('renders blowout options on third screen and calls dispatch when saved', () => { + props = { + ...props, + state: { + ...props.state, + transferType: 'distribute', + path: 'multiDispense', + disposalVolume: 20, + blowOut: 'source_well', + }, + } + render(props) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + fireEvent.click(continueBtn) + screen.getByText('Source well') + const saveBtn = screen.getByTestId('ChildNavigation_Primary_Button') + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() + }) +}) diff --git a/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/QuickTransferAdvancedSettings.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/QuickTransferAdvancedSettings.test.tsx new file mode 100644 index 00000000000..16e9012b0fc --- /dev/null +++ b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/QuickTransferAdvancedSettings.test.tsx @@ -0,0 +1,476 @@ +import type * as React from 'react' +import { fireEvent, screen } from '@testing-library/react' +import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' + +import { renderWithProviders } from '/app/__testing-utils__' +import { i18n } from '../../../../i18n' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' +import { QuickTransferAdvancedSettings } from '../../QuickTransferAdvancedSettings/' +import { useToaster } from '/app/organisms/ToasterOven' +import { PipettePath } from '../../QuickTransferAdvancedSettings/PipettePath' +import { FlowRateEntry } from '../../QuickTransferAdvancedSettings/FlowRate' +import { TipPositionEntry } from '../../QuickTransferAdvancedSettings/TipPosition' +import { Mix } from '../../QuickTransferAdvancedSettings/Mix' +import { Delay } from '../../QuickTransferAdvancedSettings/Delay' +import { TouchTip } from '../../QuickTransferAdvancedSettings/TouchTip' +import { AirGap } from '../../QuickTransferAdvancedSettings/AirGap' +import { BlowOut } from '../../QuickTransferAdvancedSettings/BlowOut' + +vi.mock('/app/redux-resources/analytics') +vi.mock('../../../../organisms/ToasterOven') +vi.mock('../../QuickTransferAdvancedSettings/PipettePath') +vi.mock('../../QuickTransferAdvancedSettings/FlowRate') +vi.mock('../../QuickTransferAdvancedSettings/TipPosition') +vi.mock('../../QuickTransferAdvancedSettings/Mix') +vi.mock('../../QuickTransferAdvancedSettings/Delay') +vi.mock('../../QuickTransferAdvancedSettings/TouchTip') +vi.mock('../../QuickTransferAdvancedSettings/AirGap') +vi.mock('../../QuickTransferAdvancedSettings/BlowOut') + +const render = ( + props: React.ComponentProps +): any => { + return renderWithProviders(, { + i18nInstance: i18n, + }) +} +let mockTrackEventWithRobotSerial: any +let mockMakeSnackbar: any + +describe('QuickTransferAdvancedSettings', () => { + let props: React.ComponentProps + + beforeEach(() => { + props = { + state: { + pipette: { + channels: 1, + liquids: [ + { + maxVolume: 1000, + minVolume: 5, + }, + ] as any, + } as any, + mount: 'left', + tipRack: { + wells: { + A1: { + totalLiquidVolume: 200, + }, + } as any, + } as any, + source: { + metadata: { + displayCategory: 'wellPlate', + }, + wells: { + A1: { + totalLiquidVolume: 200, + depth: 50, + }, + } as any, + } as any, + sourceWells: ['A1'], + destination: { + metadata: { + displayCategory: 'wellPlate', + }, + wells: { + A1: { + totalLiquidVolume: 200, + depth: 200, + }, + } as any, + } as any, + destinationWells: ['A1'], + transferType: 'consolidate', + volume: 20, + aspirateFlowRate: 570, + dispenseFlowRate: 890, + path: 'single', + tipPositionAspirate: 10, + preWetTip: false, + tipPositionDispense: 2, + changeTip: 'once', + dropTipLocation: { + cutoutId: 'cutoutA3', + cutoutFixtureId: 'trashBinAdapter', + }, + } as any, + dispatch: vi.fn(), + } + mockTrackEventWithRobotSerial = vi.fn( + () => new Promise(resolve => resolve({})) + ) + mockMakeSnackbar = vi.fn() + vi.mocked(useTrackEventWithRobotSerial).mockReturnValue({ + trackEventWithRobotSerial: mockTrackEventWithRobotSerial, + }) + vi.mocked(useToaster).mockReturnValue({ + makeSnackbar: mockMakeSnackbar, + } as any) + }) + afterEach(() => { + vi.resetAllMocks() + }) + + // base settings section + it('renders base setting options and their values', () => { + render(props) + screen.getByText('Aspirate flow rate') + screen.getByText('570 µL/s') + screen.getByText('Dispense flow rate') + screen.getByText('890 µL/s') + screen.getByText('Pipette path') + screen.getByText('Single transfers') + }) + it('renders Aspirate flow rate component when seleted', () => { + render(props) + const aspirateFlowRate = screen.getByText('Aspirate flow rate') + fireEvent.click(aspirateFlowRate) + expect(vi.mocked(FlowRateEntry)).toHaveBeenCalled() + }) + it('renders Dispense flow rate component when seleted', () => { + render(props) + const dispenseFlowRate = screen.getByText('Dispense flow rate') + fireEvent.click(dispenseFlowRate) + expect(vi.mocked(FlowRateEntry)).toHaveBeenCalled() + }) + it('renders Pipette path component when seleted', () => { + render(props) + const pipettePath = screen.getByText('Pipette path') + fireEvent.click(pipettePath) + expect(vi.mocked(PipettePath)).toHaveBeenCalled() + }) + it('Pipette path button is disabled if 1 to 1 transfer', () => { + props = { + ...props, + state: { + ...props.state, + transferType: 'transfer', + }, + } + render(props) + const pipettePath = screen.getByText('Pipette path') + fireEvent.click(pipettePath) + expect(vi.mocked(PipettePath)).not.toHaveBeenCalled() + expect(mockMakeSnackbar).toHaveBeenCalled() + }) + it('shows additional information for multi dispense', () => { + props = { + ...props, + state: { + ...props.state, + transferType: 'distribute', + path: 'multiDispense', + blowOut: 'dest_well', + disposalVolume: 40, + }, + } + render(props) + screen.getByText('Pipette path') + screen.getByText( + 'Multi-dispense, 40 µL disposal volume, blowout into destination well' + ) + }) + + // aspirate/dispense default settings section + it('renders default aspirate and dispense setting options from summary state', () => { + render(props) + screen.getByText('Aspirate Settings') + expect(screen.getAllByText('Tip position')).toHaveLength(2) + screen.getByText('10 mm from the bottom') + screen.getByText('Pre-wet tip') + expect(screen.getAllByText('Mix')).toHaveLength(2) + expect(screen.getAllByText('Delay')).toHaveLength(2) + expect(screen.getAllByText('Touch tip')).toHaveLength(2) + expect(screen.getAllByText('Air gap')).toHaveLength(2) + screen.getByText('Dispense Settings') + screen.getByText('2 mm from the bottom') + screen.getByText('Blowout') + expect(screen.getAllByText('Disabled')).toHaveLength(10) + }) + + // aspirate settings + it('opens aspirate tip position when pressed', () => { + render(props) + const aspirateTipPosition = screen.getAllByText('Tip position')[0] + fireEvent.click(aspirateTipPosition) + expect(vi.mocked(TipPositionEntry)).toHaveBeenCalled() + }) + it('renders enabled when pre-wet tip is turned on and calls dispatch when pressed', () => { + props = { + ...props, + state: { + ...props.state, + preWetTip: true, + }, + } + render(props) + const preWetTip = screen.getByText('Pre-wet tip') + expect(screen.getAllByText('Disabled')).toHaveLength(9) + screen.getByText('Enabled') + fireEvent.click(preWetTip) + expect(props.dispatch).toHaveBeenCalled() + }) + it('renders aspirate mix text when setting has value and opens mix for 1 to 1 transfer', () => { + props = { + ...props, + state: { + ...props.state, + transferType: 'transfer', + mixOnAspirate: { + mixVolume: 15, + repititions: 25, + }, + }, + } + render(props) + const mixAspirate = screen.getAllByText('Mix')[0] + screen.getByText('15 µL, 25 times') + fireEvent.click(mixAspirate) + expect(vi.mocked(Mix)).toHaveBeenCalled() + }) + it('opens aspirate mix for distribute', () => { + props = { + ...props, + state: { + ...props.state, + transferType: 'distribute', + }, + } + render(props) + const mixAspirate = screen.getAllByText('Mix')[0] + fireEvent.click(mixAspirate) + expect(vi.mocked(Mix)).toHaveBeenCalled() + }) + it('does not open aspirate mix for consolidate', () => { + render(props) + const mixAspirate = screen.getAllByText('Mix')[0] + fireEvent.click(mixAspirate) + expect(vi.mocked(Mix)).not.toHaveBeenCalled() + expect(mockMakeSnackbar).toHaveBeenCalled() + }) + it('renders aspirate delay text if there is a value in state and opens delay component when pressed', () => { + props = { + ...props, + state: { + ...props.state, + delayAspirate: { + delayDuration: 5, + positionFromBottom: 17, + }, + }, + } + render(props) + const delayAspirate = screen.getAllByText('Delay')[0] + screen.getByText('5s, 17 mm from bottom') + fireEvent.click(delayAspirate) + expect(vi.mocked(Delay)).toHaveBeenCalled() + }) + it('renders aspirate touch tip text and opens component if labware is not a reservoir', () => { + props = { + ...props, + state: { + ...props.state, + touchTipAspirate: 8, + }, + } + render(props) + const touchtipAspirate = screen.getAllByText('Touch tip')[0] + screen.getByText('8 mm from bottom') + fireEvent.click(touchtipAspirate) + expect(vi.mocked(TouchTip)).toHaveBeenCalled() + }) + it('does not open aspirate touch tip component if source labware is a reservoir', () => { + props = { + ...props, + state: { + ...props.state, + source: { + ...props.state.source, + metadata: { + displayCategory: 'reservoir', + } as any, + }, + }, + } + render(props) + const touchtipAspirate = screen.getAllByText('Touch tip')[0] + fireEvent.click(touchtipAspirate) + expect(vi.mocked(TouchTip)).not.toHaveBeenCalled() + expect(mockMakeSnackbar).toHaveBeenCalled() + }) + it('renders aspirate air gap value if it exists and opens air gap component when pressed', () => { + props = { + ...props, + state: { + ...props.state, + airGapAspirate: 2, + }, + } + render(props) + const airGapAspirate = screen.getAllByText('Air gap')[0] + screen.getByText('2 µL') + fireEvent.click(airGapAspirate) + expect(vi.mocked(AirGap)).toHaveBeenCalled() + }) + + // dispense settings + it('opens dispense tip position when pressed', () => { + render(props) + const aspirateTipPosition = screen.getAllByText('Tip position')[1] + fireEvent.click(aspirateTipPosition) + expect(vi.mocked(TipPositionEntry)).toHaveBeenCalled() + }) + it('renders dispense mix text when setting has value and opens mix for 1 to 1 transfer', () => { + props = { + ...props, + state: { + ...props.state, + transferType: 'transfer', + mixOnDispense: { + mixVolume: 18, + repititions: 20, + }, + }, + } + render(props) + const mixDispense = screen.getAllByText('Mix')[1] + screen.getByText('18 µL, 20 times') + fireEvent.click(mixDispense) + expect(vi.mocked(Mix)).toHaveBeenCalled() + }) + it('opens dispense mix for consolidate', () => { + render(props) + const mixDispense = screen.getAllByText('Mix')[1] + fireEvent.click(mixDispense) + expect(vi.mocked(Mix)).toHaveBeenCalled() + }) + it('does not open dispense mix for distribute', () => { + props = { + ...props, + state: { + ...props.state, + transferType: 'distribute', + }, + } + render(props) + const mixDispense = screen.getAllByText('Mix')[1] + fireEvent.click(mixDispense) + expect(vi.mocked(Mix)).not.toHaveBeenCalled() + expect(mockMakeSnackbar).toHaveBeenCalled() + }) + it('renders dispense delay text if there is a value in state and opens delay component when pressed', () => { + props = { + ...props, + state: { + ...props.state, + delayDispense: { + delayDuration: 10, + positionFromBottom: 4, + }, + }, + } + render(props) + const delayDispense = screen.getAllByText('Delay')[1] + screen.getByText('10s, 4 mm from bottom') + fireEvent.click(delayDispense) + expect(vi.mocked(Delay)).toHaveBeenCalled() + }) + it('renders dispense touch tip text and opens component if labware is not a reservoir', () => { + props = { + ...props, + state: { + ...props.state, + touchTipDispense: 1, + }, + } + render(props) + const touchtipDispense = screen.getAllByText('Touch tip')[1] + screen.getByText('1 mm from bottom') + fireEvent.click(touchtipDispense) + expect(vi.mocked(TouchTip)).toHaveBeenCalled() + }) + it('does not open dispense touch tip component if destination labware is a reservoir', () => { + props = { + ...props, + state: { + ...props.state, + destination: { + metadata: { + displayCategory: 'reservoir', + } as any, + } as any, + }, + } + render(props) + const touchtipDispense = screen.getAllByText('Touch tip')[1] + fireEvent.click(touchtipDispense) + expect(vi.mocked(TouchTip)).not.toHaveBeenCalled() + expect(mockMakeSnackbar).toHaveBeenCalled() + }) + it('renders dispense air gap value if it exists and opens air gap component when pressed', () => { + props = { + ...props, + state: { + ...props.state, + airGapDispense: 9, + }, + } + render(props) + const airGapDispense = screen.getAllByText('Air gap')[1] + screen.getByText('9 µL') + fireEvent.click(airGapDispense) + expect(vi.mocked(AirGap)).toHaveBeenCalled() + }) + it('renders blowout location if it is a well and opens component when clicked if transfer type', () => { + props = { + ...props, + state: { + ...props.state, + transferType: 'transfer', + blowOut: 'source_well', + }, + } + render(props) + const blowOut = screen.getByText('Blowout') + screen.getByText('Into source well') + fireEvent.click(blowOut) + expect(vi.mocked(BlowOut)).toHaveBeenCalled() + }) + it('renders blowout location if it is a trash bin and opens component when clicked if consolidate type', () => { + props = { + ...props, + state: { + ...props.state, + blowOut: { + cutoutId: 'cutoutA3', + cutoutFixtureId: 'trashBinAdapter', + }, + }, + } + render(props) + const blowOut = screen.getByText('Blowout') + screen.getByText('Into trash bin') + fireEvent.click(blowOut) + expect(vi.mocked(BlowOut)).toHaveBeenCalled() + }) + it('does not render text or open blowout component when clicked if distribute type', () => { + props = { + ...props, + state: { + ...props.state, + blowOut: 'source_well', + transferType: 'distribute', + }, + } + render(props) + const blowOut = screen.getByText('Blowout') + expect(screen.getAllByText('Disabled')).toHaveLength(10) + fireEvent.click(blowOut) + expect(vi.mocked(BlowOut)).not.toHaveBeenCalled() + expect(mockMakeSnackbar).toHaveBeenCalled() + }) +}) diff --git a/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/TipPosition.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/TipPosition.test.tsx new file mode 100644 index 00000000000..cdc4b94a735 --- /dev/null +++ b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/TipPosition.test.tsx @@ -0,0 +1,177 @@ +import type * as React from 'react' +import { fireEvent, screen } from '@testing-library/react' +import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' + +import { InputField } from '@opentrons/components' + +import { renderWithProviders } from '../../../../__testing-utils__' +import { i18n } from '../../../../i18n' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' +import { TipPositionEntry } from '../../QuickTransferAdvancedSettings/TipPosition' +import type { QuickTransferSummaryState } from '../../types' + +vi.mock('/app/redux-resources/analytics') +vi.mock('../utils') + +vi.mock('@opentrons/components', async importOriginal => { + const actualComponents = await importOriginal() + return { + ...actualComponents, + InputField: vi.fn(), + } +}) + +const render = (props: React.ComponentProps) => { + return renderWithProviders(, { + i18nInstance: i18n, + }) +} +let mockTrackEventWithRobotSerial: any + +describe('TipPosition', () => { + let props: React.ComponentProps + + beforeEach(() => { + props = { + onBack: vi.fn(), + kind: 'aspirate', + state: { + mount: 'left', + pipette: { + channels: 1, + liquids: [ + { + maxVolume: 1000, + minVolume: 5, + }, + ] as any, + } as any, + source: { + wells: { + A1: { + totalLiquidVolume: 200, + depth: 50, + }, + } as any, + } as any, + destination: { + wells: { + A1: { + totalLiquidVolume: 200, + depth: 200, + }, + } as any, + } as any, + sourceWells: ['A1'], + destinationWells: ['A1'], + transferType: 'transfer', + volume: 20, + path: 'single', + tipPositionAspirate: 10, + tipPositionDispense: 75, + } as QuickTransferSummaryState, + dispatch: vi.fn(), + } + mockTrackEventWithRobotSerial = vi.fn( + () => new Promise(resolve => resolve({})) + ) + vi.mocked(useTrackEventWithRobotSerial).mockReturnValue({ + trackEventWithRobotSerial: mockTrackEventWithRobotSerial, + }) + }) + afterEach(() => { + vi.resetAllMocks() + }) + + it('renders the tip position aspirate screen, continue, and back buttons', () => { + render(props) + screen.getByText('Aspirate tip position') + screen.getByTestId('ChildNavigation_Primary_Button') + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Distance from bottom of well (mm)', + error: null, + readOnly: true, + type: 'text', + value: 10, + }, + {} + ) + const exitBtn = screen.getByTestId('ChildNavigation_Back_Button') + fireEvent.click(exitBtn) + expect(props.onBack).toHaveBeenCalled() + }) + + it('renders the tip position dispense screen', () => { + props = { + ...props, + kind: 'dispense', + } + render(props) + screen.getByText('Dispense tip position') + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Distance from bottom of well (mm)', + error: null, + readOnly: true, + type: 'text', + value: 75, + }, + {} + ) + }) + + it('renders correct range if you enter incorrect value for aspirate', () => { + render(props) + const deleteBtn = screen.getByText('del') + fireEvent.click(deleteBtn) + fireEvent.click(deleteBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Distance from bottom of well (mm)', + error: 'Value must be between 1-100', + readOnly: true, + type: 'text', + value: 0, + }, + {} + ) + const saveBtn = screen.getByTestId('ChildNavigation_Primary_Button') + expect(saveBtn).toBeDisabled() + }) + + it('renders correct range if you enter incorrect value for dispense', () => { + props = { + ...props, + kind: 'dispense', + } + render(props) + const deleteBtn = screen.getByText('del') + fireEvent.click(deleteBtn) + fireEvent.click(deleteBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Distance from bottom of well (mm)', + error: 'Value must be between 1-400', + readOnly: true, + type: 'text', + value: 0, + }, + {} + ) + const saveBtn = screen.getByTestId('ChildNavigation_Primary_Button') + expect(saveBtn).toBeDisabled() + }) + + it('calls dispatch when an in range value is entered and saved', () => { + render(props) + const deleteBtn = screen.getByText('del') + fireEvent.click(deleteBtn) + const numButton = screen.getByText('1') + fireEvent.click(numButton) + const saveBtn = screen.getByText('Save') + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() + }) +}) diff --git a/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/TouchTip.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/TouchTip.test.tsx new file mode 100644 index 00000000000..3996d79a780 --- /dev/null +++ b/app/src/organisms/QuickTransferFlow/__tests__/QuickTransferAdvancedSettings/TouchTip.test.tsx @@ -0,0 +1,241 @@ +import type * as React from 'react' +import { fireEvent, screen } from '@testing-library/react' +import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' + +import { InputField } from '@opentrons/components' + +import { renderWithProviders } from '/app/__testing-utils__' +import { i18n } from '../../../../i18n' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' +import { TouchTip } from '../../QuickTransferAdvancedSettings/TouchTip' +import type { QuickTransferSummaryState } from '../../types' + +vi.mock('/app/redux-resources/analytics') +vi.mock('../utils') + +vi.mock('@opentrons/components', async importOriginal => { + const actualComponents = await importOriginal() + return { + ...actualComponents, + InputField: vi.fn(), + } +}) + +const render = (props: React.ComponentProps) => { + return renderWithProviders(, { + i18nInstance: i18n, + }) +} +let mockTrackEventWithRobotSerial: any + +describe('TouchTip', () => { + let props: React.ComponentProps + + beforeEach(() => { + props = { + onBack: vi.fn(), + kind: 'aspirate', + state: { + mount: 'left', + pipette: { + channels: 1, + liquids: [ + { + maxVolume: 1000, + minVolume: 5, + }, + ] as any, + } as any, + source: { + wells: { + A1: { + totalLiquidVolume: 200, + depth: 50, + }, + } as any, + } as any, + destination: { + wells: { + A1: { + totalLiquidVolume: 200, + depth: 200, + }, + } as any, + } as any, + sourceWells: ['A1'], + destinationWells: ['A1'], + transferType: 'transfer', + volume: 20, + path: 'single', + } as QuickTransferSummaryState, + dispatch: vi.fn(), + } + mockTrackEventWithRobotSerial = vi.fn( + () => new Promise(resolve => resolve({})) + ) + vi.mocked(useTrackEventWithRobotSerial).mockReturnValue({ + trackEventWithRobotSerial: mockTrackEventWithRobotSerial, + }) + }) + afterEach(() => { + vi.resetAllMocks() + }) + + it('renders the first touch tip screen, continue, and back buttons', () => { + render(props) + screen.getByText('Touch tip before aspirating') + screen.getByTestId('ChildNavigation_Primary_Button') + screen.getByText('Enabled') + screen.getByText('Disabled') + const exitBtn = screen.getByTestId('ChildNavigation_Back_Button') + fireEvent.click(exitBtn) + expect(props.onBack).toHaveBeenCalled() + }) + + it('renders the correct copy for dispense', () => { + props = { + ...props, + kind: 'dispense', + } + render(props) + screen.getByText('Touch tip before dispensing') + }) + + it('renders save button if you select enabled, then moves to second screen', () => { + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Touch tip position from bottom of well (mm)', + error: null, + readOnly: true, + type: 'number', + value: null, + }, + {} + ) + }) + + it('calls dispatch button if you select disabled and save', () => { + render(props) + const disabledBtn = screen.getByText('Disabled') + fireEvent.click(disabledBtn) + const saveBtn = screen.getByText('Save') + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() + }) + + it('has correct range for aspirate', () => { + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const numButton = screen.getByText('0') + fireEvent.click(numButton) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Touch tip position from bottom of well (mm)', + error: 'Value must be between 25-50', + readOnly: true, + type: 'number', + value: 0, + }, + {} + ) + const saveBtn = screen.getByTestId('ChildNavigation_Primary_Button') + expect(saveBtn).toBeDisabled() + }) + + it('has correct range for dispense', () => { + props = { + ...props, + kind: 'dispense', + } + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const numButton = screen.getByText('0') + fireEvent.click(numButton) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Touch tip position from bottom of well (mm)', + error: 'Value must be between 100-200', + readOnly: true, + type: 'number', + value: 0, + }, + {} + ) + const saveBtn = screen.getByTestId('ChildNavigation_Primary_Button') + expect(saveBtn).toBeDisabled() + }) + + it('calls dispatch when an in range value is entered and saved', () => { + render(props) + const enabledBtn = screen.getByText('Enabled') + fireEvent.click(enabledBtn) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + const numButton = screen.getByText('4') + fireEvent.click(numButton) + fireEvent.click(numButton) + const saveBtn = screen.getByText('Save') + fireEvent.click(saveBtn) + expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() + }) + + it('renders previously set value saved in state for aspirate', () => { + props = { + ...props, + state: { + ...props.state, + touchTipAspirate: 32, + }, + } + render(props) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Touch tip position from bottom of well (mm)', + error: null, + readOnly: true, + type: 'number', + value: 32, + }, + {} + ) + }) + + it('renders previously set value saved in state for dispense', () => { + props = { + ...props, + kind: 'dispense', + state: { + ...props.state, + touchTipDispense: 118, + }, + } + render(props) + const continueBtn = screen.getByText('Continue') + fireEvent.click(continueBtn) + expect(vi.mocked(InputField)).toHaveBeenCalledWith( + { + title: 'Touch tip position from bottom of well (mm)', + error: null, + readOnly: true, + type: 'number', + value: 118, + }, + {} + ) + }) +}) diff --git a/app/src/organisms/QuickTransferFlow/__tests__/SelectDestLabware.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/SelectDestLabware.test.tsx index 848921eb76e..6448542c14e 100644 --- a/app/src/organisms/QuickTransferFlow/__tests__/SelectDestLabware.test.tsx +++ b/app/src/organisms/QuickTransferFlow/__tests__/SelectDestLabware.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/QuickTransferFlow/__tests__/SelectPipette.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/SelectPipette.test.tsx index 463b2b7bb61..e32f7645593 100644 --- a/app/src/organisms/QuickTransferFlow/__tests__/SelectPipette.test.tsx +++ b/app/src/organisms/QuickTransferFlow/__tests__/SelectPipette.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' import { useInstrumentsQuery } from '@opentrons/react-api-client' diff --git a/app/src/organisms/QuickTransferFlow/__tests__/SelectSourceLabware.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/SelectSourceLabware.test.tsx index 1f353557e1c..73121ea7b2d 100644 --- a/app/src/organisms/QuickTransferFlow/__tests__/SelectSourceLabware.test.tsx +++ b/app/src/organisms/QuickTransferFlow/__tests__/SelectSourceLabware.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/QuickTransferFlow/__tests__/SelectTipRack.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/SelectTipRack.test.tsx index 39f9f7cb5fb..f4ee22bd2b9 100644 --- a/app/src/organisms/QuickTransferFlow/__tests__/SelectTipRack.test.tsx +++ b/app/src/organisms/QuickTransferFlow/__tests__/SelectTipRack.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/QuickTransferFlow/__tests__/SummaryAndSettings.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/SummaryAndSettings.test.tsx index ff04b25c6e0..0f5f7c7742c 100644 --- a/app/src/organisms/QuickTransferFlow/__tests__/SummaryAndSettings.test.tsx +++ b/app/src/organisms/QuickTransferFlow/__tests__/SummaryAndSettings.test.tsx @@ -1,14 +1,18 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' + import { useCreateProtocolMutation, useCreateRunMutation, } from '@opentrons/react-api-client' + import { useNotifyDeckConfigurationQuery } from '/app/resources/deck_configuration' +import { ANALYTICS_QUICK_TRANSFER_RUN_NOW } from '/app/redux/analytics' import { createQuickTransferFile } from '../utils' import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' import { SummaryAndSettings } from '../SummaryAndSettings' import { NameQuickTransfer } from '../NameQuickTransfer' import { Overview } from '../Overview' @@ -25,6 +29,7 @@ vi.mock('react-router-dom', async importOriginal => { }) vi.mock('../Overview') vi.mock('../NameQuickTransfer') +vi.mock('/app/redux-resources/analytics') vi.mock('../utils', async () => { const actual = await vi.importActual('../utils') return { @@ -41,6 +46,7 @@ const render = (props: React.ComponentProps) => { i18nInstance: i18n, }) } +let mockTrackEventWithRobotSerial: any describe('SummaryAndSettings', () => { let props: React.ComponentProps @@ -65,13 +71,19 @@ describe('SummaryAndSettings', () => { transferType: 'transfer', volume: 25, }, + analyticsStartTime: new Date(), } + mockTrackEventWithRobotSerial = vi.fn( + () => new Promise(resolve => resolve({})) + ) vi.mocked(useNotifyDeckConfigurationQuery).mockReturnValue({ data: { data: [], }, } as any) - + vi.mocked(useTrackEventWithRobotSerial).mockReturnValue({ + trackEventWithRobotSerial: mockTrackEventWithRobotSerial, + }) vi.mocked(useCreateProtocolMutation).mockReturnValue({ mutateAsync: createProtocol, } as any) @@ -112,6 +124,7 @@ describe('SummaryAndSettings', () => { render(props) const continueBtn = screen.getByTestId('ChildNavigation_Primary_Button') fireEvent.click(continueBtn) + expect(mockTrackEventWithRobotSerial).toHaveBeenCalled() screen.getByText('Do you want to run your quick transfer now?') screen.getByText('Save your quick transfer to run it in the future.') }) @@ -129,6 +142,10 @@ describe('SummaryAndSettings', () => { fireEvent.click(continueBtn) const runBtn = screen.getByText('Run now') fireEvent.click(runBtn) + expect(mockTrackEventWithRobotSerial).toHaveBeenCalledWith({ + name: ANALYTICS_QUICK_TRANSFER_RUN_NOW, + properties: {}, + }) expect(vi.mocked(createQuickTransferFile)).toHaveBeenCalled() expect(vi.mocked(createProtocol)).toHaveBeenCalled() }) diff --git a/app/src/organisms/QuickTransferFlow/__tests__/TipManagement/ChangeTip.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/TipManagement/ChangeTip.test.tsx index 2ca682f75f0..213633678e5 100644 --- a/app/src/organisms/QuickTransferFlow/__tests__/TipManagement/ChangeTip.test.tsx +++ b/app/src/organisms/QuickTransferFlow/__tests__/TipManagement/ChangeTip.test.tsx @@ -1,17 +1,23 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' +import { ANALYTICS_QUICK_TRANSFER_SETTING_SAVED } from '/app/redux/analytics' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' import { ChangeTip } from '../../TipManagement/ChangeTip' +vi.mock('/app/redux-resources/analytics') + const render = (props: React.ComponentProps): any => { return renderWithProviders(, { i18nInstance: i18n, }) } +let mockTrackEventWithRobotSerial: any + describe('ChangeTip', () => { let props: React.ComponentProps @@ -28,6 +34,12 @@ describe('ChangeTip', () => { } as any, dispatch: vi.fn(), } + mockTrackEventWithRobotSerial = vi.fn( + () => new Promise(resolve => resolve({})) + ) + vi.mocked(useTrackEventWithRobotSerial).mockReturnValue({ + trackEventWithRobotSerial: mockTrackEventWithRobotSerial, + }) }) afterEach(() => { vi.resetAllMocks() @@ -49,6 +61,10 @@ describe('ChangeTip', () => { const saveBtn = screen.getByText('Save') fireEvent.click(saveBtn) expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalledWith({ + name: ANALYTICS_QUICK_TRANSFER_SETTING_SAVED, + properties: { setting: 'ChangeTip' }, + }) }) it('renders correct change tip options when single transfer of less than 96 wells', () => { render(props) diff --git a/app/src/organisms/QuickTransferFlow/__tests__/TipManagement/TipDropLocation.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/TipManagement/TipDropLocation.test.tsx index 92d119a80c6..aed3d143b31 100644 --- a/app/src/organisms/QuickTransferFlow/__tests__/TipManagement/TipDropLocation.test.tsx +++ b/app/src/organisms/QuickTransferFlow/__tests__/TipManagement/TipDropLocation.test.tsx @@ -1,19 +1,23 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' import { useNotifyDeckConfigurationQuery } from '/app/resources/deck_configuration' +import { ANALYTICS_QUICK_TRANSFER_SETTING_SAVED } from '/app/redux/analytics' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' import { TipDropLocation } from '../../TipManagement/TipDropLocation' vi.mock('/app/resources/deck_configuration') +vi.mock('/app/redux-resources/analytics') const render = (props: React.ComponentProps): any => { return renderWithProviders(, { i18nInstance: i18n, }) } +let mockTrackEventWithRobotSerial: any describe('TipDropLocation', () => { let props: React.ComponentProps @@ -26,6 +30,12 @@ describe('TipDropLocation', () => { } as any, dispatch: vi.fn(), } + mockTrackEventWithRobotSerial = vi.fn( + () => new Promise(resolve => resolve({})) + ) + vi.mocked(useTrackEventWithRobotSerial).mockReturnValue({ + trackEventWithRobotSerial: mockTrackEventWithRobotSerial, + }) vi.mocked(useNotifyDeckConfigurationQuery).mockReturnValue({ data: [ { @@ -62,5 +72,9 @@ describe('TipDropLocation', () => { const saveBtn = screen.getByText('Save') fireEvent.click(saveBtn) expect(props.dispatch).toHaveBeenCalled() + expect(mockTrackEventWithRobotSerial).toHaveBeenCalledWith({ + name: ANALYTICS_QUICK_TRANSFER_SETTING_SAVED, + properties: { setting: 'TipDropLocation' }, + }) }) }) diff --git a/app/src/organisms/QuickTransferFlow/__tests__/TipManagement/TipManagement.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/TipManagement/TipManagement.test.tsx index 79f50c4c09b..618153a8b53 100644 --- a/app/src/organisms/QuickTransferFlow/__tests__/TipManagement/TipManagement.test.tsx +++ b/app/src/organisms/QuickTransferFlow/__tests__/TipManagement/TipManagement.test.tsx @@ -1,21 +1,25 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' +import { ANALYTICS_QUICK_TRANSFER_TIP_MANAGEMENT_TAB } from '/app/redux/analytics' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' import { ChangeTip } from '../../TipManagement/ChangeTip' import { TipDropLocation } from '../../TipManagement/TipDropLocation' import { TipManagement } from '../../TipManagement/' vi.mock('../../TipManagement/ChangeTip') vi.mock('../../TipManagement/TipDropLocation') +vi.mock('/app/redux-resources/analytics') const render = (props: React.ComponentProps): any => { return renderWithProviders(, { i18nInstance: i18n, }) } +let mockTrackEventWithRobotSerial: any describe('TipManagement', () => { let props: React.ComponentProps @@ -30,6 +34,12 @@ describe('TipManagement', () => { } as any, dispatch: vi.fn(), } + mockTrackEventWithRobotSerial = vi.fn( + () => new Promise(resolve => resolve({})) + ) + vi.mocked(useTrackEventWithRobotSerial).mockReturnValue({ + trackEventWithRobotSerial: mockTrackEventWithRobotSerial, + }) }) afterEach(() => { vi.resetAllMocks() @@ -41,6 +51,10 @@ describe('TipManagement', () => { screen.getByText('Once at the start of the transfer') screen.getByText('Tip drop location') screen.getByText('Trash bin') + expect(mockTrackEventWithRobotSerial).toHaveBeenCalledWith({ + name: ANALYTICS_QUICK_TRANSFER_TIP_MANAGEMENT_TAB, + properties: {}, + }) }) it('renders Change tip component when seleted', () => { render(props) diff --git a/app/src/organisms/QuickTransferFlow/__tests__/VolumeEntry.test.tsx b/app/src/organisms/QuickTransferFlow/__tests__/VolumeEntry.test.tsx index 007adf2e98e..8a14b9a5993 100644 --- a/app/src/organisms/QuickTransferFlow/__tests__/VolumeEntry.test.tsx +++ b/app/src/organisms/QuickTransferFlow/__tests__/VolumeEntry.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, afterEach, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/QuickTransferFlow/index.tsx b/app/src/organisms/QuickTransferFlow/index.tsx index 149c52b3c07..f7d94a46000 100644 --- a/app/src/organisms/QuickTransferFlow/index.tsx +++ b/app/src/organisms/QuickTransferFlow/index.tsx @@ -6,6 +6,8 @@ import { StepMeter, POSITION_STICKY, } from '@opentrons/components' +import { ANALYTICS_QUICK_TRANSFER_EXIT_EARLY } from '/app/redux/analytics' +import { useTrackEventWithRobotSerial } from '/app/redux-resources/analytics' import { ConfirmExitModal } from './ConfirmExitModal' import { CreateNewTransfer } from './CreateNewTransfer' import { SelectPipette } from './SelectPipette' @@ -27,17 +29,26 @@ const initialQuickTransferState: QuickTransferWizardState = {} export const QuickTransferFlow = (): JSX.Element => { const navigate = useNavigate() const { i18n, t } = useTranslation(['quick_transfer', 'shared']) + const { trackEventWithRobotSerial } = useTrackEventWithRobotSerial() const [state, dispatch] = React.useReducer( quickTransferWizardReducer, initialQuickTransferState ) const [currentStep, setCurrentStep] = React.useState(0) + const [analyticsStartTime] = React.useState(new Date()) + const { confirm: confirmExit, showConfirmation: showConfirmExit, cancel: cancelExit, } = useConditionalConfirm(() => { + trackEventWithRobotSerial({ + name: ANALYTICS_QUICK_TRANSFER_EXIT_EARLY, + properties: { + step: currentStep, + }, + }) navigate('/quick-transfer') }, true) @@ -73,7 +84,11 @@ export const QuickTransferFlow = (): JSX.Element => { , , , - , + , ] return ( diff --git a/app/src/organisms/QuickTransferFlow/utils/createQuickTransferFile.ts b/app/src/organisms/QuickTransferFlow/utils/createQuickTransferFile.ts index a4114c0a4d4..09fa2348f39 100644 --- a/app/src/organisms/QuickTransferFlow/utils/createQuickTransferFile.ts +++ b/app/src/organisms/QuickTransferFlow/utils/createQuickTransferFile.ts @@ -202,10 +202,9 @@ export function createQuickTransferFile( subcategory: null, tags: [], }, - // TODO: formalize designer application data type designerApplication: { - name: 'Quick Transfer', - version: '0.0', + name: 'opentrons/quick-transfer', + version: '1.0.0', data: quickTransferState, }, } @@ -250,8 +249,6 @@ export function createQuickTransferFile( ...commandAnnotionaV1Mixin, }) - // Leaving this in for debugging while work is still in flight - console.log('Here are the protocol contents:', protocolContents) return new File( [protocolContents], `${protocolBase.metadata.protocolName}.json` diff --git a/app/src/organisms/RobotSettingsCalibration/CalibrationDataDownload.tsx b/app/src/organisms/RobotSettingsCalibration/CalibrationDataDownload.tsx index 5aae1a1425f..7bff8bdf29f 100644 --- a/app/src/organisms/RobotSettingsCalibration/CalibrationDataDownload.tsx +++ b/app/src/organisms/RobotSettingsCalibration/CalibrationDataDownload.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { saveAs } from 'file-saver' import { useTranslation, Trans } from 'react-i18next' diff --git a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/ModuleCalibrationItems.tsx b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/ModuleCalibrationItems.tsx index faae54b48d3..11ee8f60402 100644 --- a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/ModuleCalibrationItems.tsx +++ b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/ModuleCalibrationItems.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import styled, { css } from 'styled-components' diff --git a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/ModuleCalibrationOverflowMenu.tsx b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/ModuleCalibrationOverflowMenu.tsx index 79aa70395ef..36ba8fee26d 100644 --- a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/ModuleCalibrationOverflowMenu.tsx +++ b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/ModuleCalibrationOverflowMenu.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { useTranslation } from 'react-i18next' @@ -19,9 +19,9 @@ import { import { useChainLiveCommands } from '/app/resources/runs' import { useRunStatuses } from '/app/organisms/Devices/hooks' -import { getModulePrepCommands } from '/app/organisms/Devices/getModulePrepCommands' +import { getModulePrepCommands } from '/app/local-resources/modules' import { ModuleWizardFlows } from '/app/organisms/ModuleWizardFlows' -import { getModuleTooHot } from '/app/organisms/Devices/getModuleTooHot' +import { getModuleTooHot } from '/app/transformations/modules' import { useIsEstopNotDisengaged } from '/app/resources/devices/hooks/useIsEstopNotDisengaged' import type { AttachedModule } from '/app/redux/modules/types' @@ -54,7 +54,7 @@ export function ModuleCalibrationOverflowMenu({ setShowOverflowMenu, } = useMenuHandleClickOutside() - const [showModuleWizard, setShowModuleWizard] = React.useState(false) + const [showModuleWizard, setShowModuleWizard] = useState(false) const { isRunRunning: isRunning } = useRunStatuses() const [targetProps, tooltipProps] = useHoverTooltip() @@ -73,7 +73,7 @@ export function ModuleCalibrationOverflowMenu({ const [ prepCommandErrorMessage, setPrepCommandErrorMessage, - ] = React.useState('') + ] = useState('') const isEstopNotDisengaged = useIsEstopNotDisengaged(robotName) @@ -87,7 +87,7 @@ export function ModuleCalibrationOverflowMenu({ setShowModuleWizard(true) } - React.useEffect(() => { + useEffect(() => { if (isRunning) { updateRobotStatus(true) } diff --git a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/PipetteOffsetCalibrationItems.tsx b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/PipetteOffsetCalibrationItems.tsx index f6bba1f6921..d94c51c681d 100644 --- a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/PipetteOffsetCalibrationItems.tsx +++ b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/PipetteOffsetCalibrationItems.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' import styled, { css } from 'styled-components' @@ -17,10 +16,8 @@ import { OverflowMenu } from './OverflowMenu' import { formatLastCalibrated, getDisplayNameForTipRack } from './utils' import { getCustomLabwareDefinitions } from '/app/redux/custom-labware' import { LEFT } from '/app/redux/pipettes' -import { - useAttachedPipettes, - useAttachedPipettesFromInstrumentsQuery, -} from '/app/organisms/Devices/hooks' +import { useAttachedPipettesFromInstrumentsQuery } from '/app/organisms/Devices/hooks' +import { useAttachedPipettes } from '/app/resources/instruments' import { useIsFlex } from '/app/redux-resources/robots' import type { State } from '/app/redux/types' diff --git a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/TipLengthCalibrationItems.tsx b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/TipLengthCalibrationItems.tsx index 5734e7180c3..9aea786c421 100644 --- a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/TipLengthCalibrationItems.tsx +++ b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/TipLengthCalibrationItems.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' import styled, { css } from 'styled-components' @@ -11,7 +10,7 @@ import { TYPOGRAPHY, } from '@opentrons/components' -import { useAttachedPipettes } from '/app/organisms/Devices/hooks' +import { useAttachedPipettes } from '/app/resources/instruments' import { getCustomLabwareDefinitions } from '/app/redux/custom-labware' import { OverflowMenu } from './OverflowMenu' import { formatLastCalibrated, getDisplayNameForTipRack } from './utils' diff --git a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/ModuleCalibrationItems.test.tsx b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/ModuleCalibrationItems.test.tsx index 77b2f3ee6ae..2fdd9694e5d 100644 --- a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/ModuleCalibrationItems.test.tsx +++ b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/ModuleCalibrationItems.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import { i18n } from '/app/i18n' diff --git a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/ModuleCalibrationOverflowMenu.test.tsx b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/ModuleCalibrationOverflowMenu.test.tsx index bf37956c884..cc7c1e847e1 100644 --- a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/ModuleCalibrationOverflowMenu.test.tsx +++ b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/ModuleCalibrationOverflowMenu.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen, waitFor } from '@testing-library/react' import { when } from 'vitest-when' import { describe, it, expect, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/OverflowMenu.test.tsx b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/OverflowMenu.test.tsx index f112040edc5..3a01e166746 100644 --- a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/OverflowMenu.test.tsx +++ b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/OverflowMenu.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { when } from 'vitest-when' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/PipetteOffsetCalibrationItems.test.tsx b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/PipetteOffsetCalibrationItems.test.tsx index 57e528d1554..2cd4c6077df 100644 --- a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/PipetteOffsetCalibrationItems.test.tsx +++ b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/PipetteOffsetCalibrationItems.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' @@ -9,10 +9,8 @@ import { mockAttachedPipette, mockAttachedPipetteInformation, } from '/app/redux/pipettes/__fixtures__' -import { - useAttachedPipettes, - useAttachedPipettesFromInstrumentsQuery, -} from '/app/organisms/Devices/hooks' +import { useAttachedPipettesFromInstrumentsQuery } from '/app/organisms/Devices/hooks' +import { useAttachedPipettes } from '/app/resources/instruments' import { useIsFlex } from '/app/redux-resources/robots' import { renderWithProviders } from '/app/__testing-utils__' import { PipetteOffsetCalibrationItems } from '../PipetteOffsetCalibrationItems' @@ -64,6 +62,7 @@ vi.mock('/app/redux/discovery') vi.mock('/app/assets/labware/findLabware') vi.mock('/app/organisms/Devices/hooks') vi.mock('/app/redux-resources/robots') +vi.mock('/app/resources/instruments') vi.mock('../OverflowMenu') const mockAttachedPipettes: AttachedPipettesByMount = { diff --git a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/TipLengthCalibrationItems.test.tsx b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/TipLengthCalibrationItems.test.tsx index 44481da6f3d..71746c541c7 100644 --- a/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/TipLengthCalibrationItems.test.tsx +++ b/app/src/organisms/RobotSettingsCalibration/CalibrationDetails/__tests__/TipLengthCalibrationItems.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/RobotSettingsCalibration/CalibrationHealthCheck.tsx b/app/src/organisms/RobotSettingsCalibration/CalibrationHealthCheck.tsx index 15592842d02..833dc965770 100644 --- a/app/src/organisms/RobotSettingsCalibration/CalibrationHealthCheck.tsx +++ b/app/src/organisms/RobotSettingsCalibration/CalibrationHealthCheck.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { createPortal } from 'react-dom' import { useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' @@ -27,12 +27,12 @@ import * as Calibration from '/app/redux/calibration' import * as Config from '/app/redux/config' import * as Pipettes from '/app/redux/pipettes' import * as Sessions from '/app/redux/sessions' +import { useRunStatuses } from '/app/organisms/Devices/hooks' import { - useDeckCalibrationStatus, useAttachedPipettes, useAttachedPipetteCalibrations, - useRunStatuses, -} from '/app/organisms/Devices/hooks' +} from '/app/resources/instruments' +import { useDeckCalibrationStatus } from '/app/resources/calibration' import type { AttachedPipettesByMount, @@ -74,7 +74,7 @@ export function CalibrationHealthCheck({ placement: TOOLTIP_LEFT, }) - const [showCalBlockModal, setShowCalBlockModal] = React.useState(false) + const [showCalBlockModal, setShowCalBlockModal] = useState(false) const deckCalibrationStatus = useDeckCalibrationStatus(robotName) const attachedPipettes = useAttachedPipettes() diff --git a/app/src/organisms/RobotSettingsCalibration/DeckCalibrationConfirmModal.tsx b/app/src/organisms/RobotSettingsCalibration/DeckCalibrationConfirmModal.tsx index c07cef244ed..215ff113fe6 100644 --- a/app/src/organisms/RobotSettingsCalibration/DeckCalibrationConfirmModal.tsx +++ b/app/src/organisms/RobotSettingsCalibration/DeckCalibrationConfirmModal.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { ALIGN_CENTER, diff --git a/app/src/organisms/RobotSettingsCalibration/RobotSettingsDeckCalibration.tsx b/app/src/organisms/RobotSettingsCalibration/RobotSettingsDeckCalibration.tsx index 6b48db62339..48a34c166fe 100644 --- a/app/src/organisms/RobotSettingsCalibration/RobotSettingsDeckCalibration.tsx +++ b/app/src/organisms/RobotSettingsCalibration/RobotSettingsDeckCalibration.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useRef, useEffect } from 'react' import { useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' @@ -26,7 +26,7 @@ export function RobotSettingsDeckCalibration({ robotName, }: RobotSettingsDeckCalibrationProps): JSX.Element { const { t } = useTranslation('device_settings') - const createRequestId = React.useRef(null) + const createRequestId = useRef(null) const robot = useRobot(robotName) const deckCalibrationData = useDeckCalibrationData(robot?.name) @@ -47,7 +47,7 @@ export function RobotSettingsDeckCalibration({ }) : t('not_calibrated') - React.useEffect(() => { + useEffect(() => { if (createStatus === RobotApi.SUCCESS) { createRequestId.current = null } diff --git a/app/src/organisms/RobotSettingsCalibration/RobotSettingsGripperCalibration.tsx b/app/src/organisms/RobotSettingsCalibration/RobotSettingsGripperCalibration.tsx index fef63c269ad..90cd9c9f4a8 100644 --- a/app/src/organisms/RobotSettingsCalibration/RobotSettingsGripperCalibration.tsx +++ b/app/src/organisms/RobotSettingsCalibration/RobotSettingsGripperCalibration.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import styled, { css } from 'styled-components' @@ -71,7 +71,7 @@ export function RobotSettingsGripperCalibration( setShowOverflowMenu(false) }, }) - const [showWizardFlow, setShowWizardFlow] = React.useState(false) + const [showWizardFlow, setShowWizardFlow] = useState(false) const isEstopNotDisengaged = useIsEstopNotDisengaged(robotName) const gripperCalibrationLastModified = diff --git a/app/src/organisms/RobotSettingsCalibration/RobotSettingsModuleCalibration.tsx b/app/src/organisms/RobotSettingsCalibration/RobotSettingsModuleCalibration.tsx index c6cec90fbba..31e0e3114aa 100644 --- a/app/src/organisms/RobotSettingsCalibration/RobotSettingsModuleCalibration.tsx +++ b/app/src/organisms/RobotSettingsCalibration/RobotSettingsModuleCalibration.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/RobotSettingsCalibration/RobotSettingsPipetteOffsetCalibration.tsx b/app/src/organisms/RobotSettingsCalibration/RobotSettingsPipetteOffsetCalibration.tsx index 78a9ee9db51..d066d2b74f0 100644 --- a/app/src/organisms/RobotSettingsCalibration/RobotSettingsPipetteOffsetCalibration.tsx +++ b/app/src/organisms/RobotSettingsCalibration/RobotSettingsPipetteOffsetCalibration.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { @@ -15,7 +14,7 @@ import { usePipetteOffsetCalibrations, } from '../Devices/hooks' import { useIsFlex } from '/app/redux-resources/robots' -import { getShowPipetteCalibrationWarning } from '../Devices/utils' +import { getShowPipetteCalibrationWarning } from '/app/transformations/instruments' import { PipetteRecalibrationWarning } from '../Devices/PipetteCard/PipetteRecalibrationWarning' import { PipetteOffsetCalibrationItems } from './CalibrationDetails/PipetteOffsetCalibrationItems' diff --git a/app/src/organisms/RobotSettingsCalibration/RobotSettingsTipLengthCalibration.tsx b/app/src/organisms/RobotSettingsCalibration/RobotSettingsTipLengthCalibration.tsx index 362bd05ed28..ebca2db82c8 100644 --- a/app/src/organisms/RobotSettingsCalibration/RobotSettingsTipLengthCalibration.tsx +++ b/app/src/organisms/RobotSettingsCalibration/RobotSettingsTipLengthCalibration.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { getLabwareDefURI } from '@opentrons/shared-data' @@ -11,7 +10,7 @@ import { } from '@opentrons/components' import { useAllTipLengthCalibrationsQuery } from '@opentrons/react-api-client' -import { useAttachedPipettes } from '/app/organisms/Devices/hooks' +import { useAttachedPipettes } from '/app/resources/instruments' import { getDefaultTiprackDefForPipetteName } from '../Devices/constants' import { TipLengthCalibrationItems } from './CalibrationDetails/TipLengthCalibrationItems' diff --git a/app/src/organisms/RobotSettingsCalibration/__tests__/CalibrationDataDownload.test.tsx b/app/src/organisms/RobotSettingsCalibration/__tests__/CalibrationDataDownload.test.tsx index cb5a764bc53..54e795a1d1e 100644 --- a/app/src/organisms/RobotSettingsCalibration/__tests__/CalibrationDataDownload.test.tsx +++ b/app/src/organisms/RobotSettingsCalibration/__tests__/CalibrationDataDownload.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { when } from 'vitest-when' import { describe, diff --git a/app/src/organisms/RobotSettingsCalibration/__tests__/CalibrationHealthCheck.test.tsx b/app/src/organisms/RobotSettingsCalibration/__tests__/CalibrationHealthCheck.test.tsx index 2fafaf3acce..3c2f2c5b3db 100644 --- a/app/src/organisms/RobotSettingsCalibration/__tests__/CalibrationHealthCheck.test.tsx +++ b/app/src/organisms/RobotSettingsCalibration/__tests__/CalibrationHealthCheck.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import userEvent from '@testing-library/user-event' import { fireEvent, screen, waitFor } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' @@ -19,11 +19,12 @@ import { mockTipLengthCalibration2, } from '/app/redux/calibration/tip-length/__fixtures__' import { mockAttachedPipette } from '/app/redux/pipettes/__fixtures__' +import { useRunStatuses } from '/app/organisms/Devices/hooks' + import { useAttachedPipettes, useAttachedPipetteCalibrations, - useRunStatuses, -} from '/app/organisms/Devices/hooks' +} from '/app/resources/instruments' import { CalibrationHealthCheck } from '../CalibrationHealthCheck' @@ -36,6 +37,8 @@ vi.mock('/app/redux/analytics') vi.mock('/app/redux/config') vi.mock('/app/redux/pipettes') vi.mock('/app/organisms/Devices/hooks') +vi.mock('/app/resources/instruments') +vi.mock('/app/redux-resources/robots') const mockAttachedPipettes: AttachedPipettesByMount = { left: mockAttachedPipette, diff --git a/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsCalibration.test.tsx b/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsCalibration.test.tsx index 03ca54a3d48..763963dfb76 100644 --- a/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsCalibration.test.tsx +++ b/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsCalibration.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { when } from 'vitest-when' import { screen } from '@testing-library/react' import { describe, it, expect, beforeEach, vi } from 'vitest' @@ -21,10 +20,10 @@ import { } from '/app/redux/pipettes/__fixtures__' import { usePipetteOffsetCalibrations, - useAttachedPipettes, useRunStatuses, useAttachedPipettesFromInstrumentsQuery, } from '/app/organisms/Devices/hooks' +import { useAttachedPipettes } from '/app/resources/instruments' import { useRobot, useIsFlex } from '/app/redux-resources/robots' import { CalibrationDataDownload } from '../CalibrationDataDownload' @@ -50,6 +49,7 @@ vi.mock('/app/redux/config') vi.mock('/app/redux/sessions/selectors') vi.mock('/app/redux/robot-api/selectors') vi.mock('/app/redux-resources/robots') +vi.mock('/app/resources/instruments') vi.mock('/app/organisms/Devices/hooks') vi.mock('../CalibrationDataDownload') vi.mock('../CalibrationHealthCheck') diff --git a/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsDeckCalibration.test.tsx b/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsDeckCalibration.test.tsx index 7f0c90b02a0..fb48e3c3ff5 100644 --- a/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsDeckCalibration.test.tsx +++ b/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsDeckCalibration.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, vi, beforeEach } from 'vitest' @@ -10,10 +10,8 @@ import { } from '/app/redux/calibration/__fixtures__' import { mockConnectableRobot } from '/app/redux/discovery/__fixtures__' import { mockAttachedPipette } from '/app/redux/pipettes/__fixtures__' -import { - useDeckCalibrationData, - useAttachedPipettes, -} from '/app/organisms/Devices/hooks' +import { useDeckCalibrationData } from '/app/organisms/Devices/hooks' +import { useAttachedPipettes } from '/app/resources/instruments' import { useRobot } from '/app/redux-resources/robots' import { renderWithProviders } from '/app/__testing-utils__' @@ -25,6 +23,7 @@ vi.mock('/app/organisms/CalibrationStatusCard') vi.mock('/app/redux/robot-api/selectors') vi.mock('/app/redux-resources/robots') vi.mock('/app/organisms/Devices/hooks') +vi.mock('/app/resources/instruments') const mockAttachedPipettes: AttachedPipettesByMount = { left: mockAttachedPipette, diff --git a/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsGripperCalibration.test.tsx b/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsGripperCalibration.test.tsx index 3e5d952ea09..e4a4d48eea1 100644 --- a/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsGripperCalibration.test.tsx +++ b/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsGripperCalibration.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' diff --git a/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsModuleCalibration.test.tsx b/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsModuleCalibration.test.tsx index 8c2f9fa9a4c..95b71a450af 100644 --- a/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsModuleCalibration.test.tsx +++ b/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsModuleCalibration.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, vi, beforeEach } from 'vitest' import { i18n } from '/app/i18n' diff --git a/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsPipetteOffsetCalibration.test.tsx b/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsPipetteOffsetCalibration.test.tsx index 5c88977ca7c..5133923e071 100644 --- a/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsPipetteOffsetCalibration.test.tsx +++ b/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsPipetteOffsetCalibration.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { describe, it, vi, beforeEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsTipLengthCalibration.test.tsx b/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsTipLengthCalibration.test.tsx index 0549050d7e1..b2934dceac4 100644 --- a/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsTipLengthCalibration.test.tsx +++ b/app/src/organisms/RobotSettingsCalibration/__tests__/RobotSettingsTipLengthCalibration.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, beforeEach, vi } from 'vitest' import { i18n } from '/app/i18n' @@ -10,10 +9,8 @@ import { mockTipLengthCalibration3, } from '/app/redux/calibration/tip-length/__fixtures__' import { mockAttachedPipette } from '/app/redux/pipettes/__fixtures__' -import { - useAttachedPipettes, - useTipLengthCalibrations, -} from '/app/organisms/Devices/hooks' +import { useTipLengthCalibrations } from '/app/organisms/Devices/hooks' +import { useAttachedPipettes } from '/app/resources/instruments' import { RobotSettingsTipLengthCalibration } from '../RobotSettingsTipLengthCalibration' import { TipLengthCalibrationItems } from '../CalibrationDetails/TipLengthCalibrationItems' @@ -24,6 +21,7 @@ import type { AttachedPipettesByMount } from '/app/redux/pipettes/types' vi.mock('/app/redux/config') vi.mock('/app/organisms/Devices/hooks') vi.mock('../CalibrationDetails/TipLengthCalibrationItems') +vi.mock('/app/resources/instruments') const mockFormattedPipetteOffsetCalibrations: FormattedPipetteOffsetCalibration[] = [] diff --git a/app/src/organisms/RobotSettingsCalibration/index.tsx b/app/src/organisms/RobotSettingsCalibration/index.tsx index 22bff478880..cc6571ccf9c 100644 --- a/app/src/organisms/RobotSettingsCalibration/index.tsx +++ b/app/src/organisms/RobotSettingsCalibration/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useRef, useState, useEffect } from 'react' import { createPortal } from 'react-dom' import { useSelector, useDispatch } from 'react-redux' import { useTranslation } from 'react-i18next' @@ -77,21 +77,21 @@ export function RobotSettingsCalibration({ 'robot_calibration', 'shared', ]) - const trackedRequestId = React.useRef(null) - const createRequestId = React.useRef(null) - const jogRequestId = React.useRef(null) + const trackedRequestId = useRef(null) + const createRequestId = useRef(null) + const jogRequestId = useRef(null) const [ showHowCalibrationWorksModal, setShowHowCalibrationWorksModal, - ] = React.useState(false) + ] = useState(false) const robot = useRobot(robotName) const notConnectable = robot?.status !== CONNECTABLE const isFlex = useIsFlex(robotName) const dispatch = useDispatch() - React.useEffect(() => { + useEffect(() => { dispatch(Sessions.fetchAllSessions(robotName)) }, [dispatch, robotName]) @@ -249,7 +249,7 @@ export function RobotSettingsCalibration({ }) } - React.useEffect(() => { + useEffect(() => { if (createStatus === RobotApi.SUCCESS) { createRequestId.current = null } diff --git a/app/src/organisms/RobotSetupHeader/index.tsx b/app/src/organisms/RobotSetupHeader/index.tsx index 3a192c28148..6b7a3fa1049 100644 --- a/app/src/organisms/RobotSetupHeader/index.tsx +++ b/app/src/organisms/RobotSetupHeader/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { ALIGN_CENTER, diff --git a/app/src/organisms/RunPreview/index.tsx b/app/src/organisms/RunPreview/index.tsx index 6cb9b50e052..a59a4e774a5 100644 --- a/app/src/organisms/RunPreview/index.tsx +++ b/app/src/organisms/RunPreview/index.tsx @@ -21,16 +21,16 @@ import { TYPOGRAPHY, } from '@opentrons/components' -import { useMostRecentCompletedAnalysis } from '../LabwarePositionCheck/useMostRecentCompletedAnalysis' import { useNotifyAllCommandsAsPreSerializedList, useNotifyRunQuery, useRunStatus, + useMostRecentCompletedAnalysis, + useLastRunCommand, } from '/app/resources/runs' import { CommandText, CommandIcon } from '/app/molecules/Command' import { Divider } from '/app/atoms/structure' import { NAV_BAR_WIDTH } from '../../App/constants' -import { useLastRunCommand } from '../Devices/hooks/useLastRunCommand' import type { RunStatus } from '@opentrons/api-client' import type { RobotType } from '@opentrons/shared-data' diff --git a/app/src/organisms/RunProgressMeter/InterventionTicks.tsx b/app/src/organisms/RunProgressMeter/InterventionTicks.tsx index f8dfd722f17..8bab7b981af 100644 --- a/app/src/organisms/RunProgressMeter/InterventionTicks.tsx +++ b/app/src/organisms/RunProgressMeter/InterventionTicks.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import last from 'lodash/last' import { Tick } from './Tick' import type { RunTimeCommand } from '@opentrons/shared-data' diff --git a/app/src/organisms/RunProgressMeter/Tick.tsx b/app/src/organisms/RunProgressMeter/Tick.tsx index 5bb644f87b8..1976e8d4553 100644 --- a/app/src/organisms/RunProgressMeter/Tick.tsx +++ b/app/src/organisms/RunProgressMeter/Tick.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { createPortal } from 'react-dom' diff --git a/app/src/organisms/RunProgressMeter/__tests__/InterventionTicks.test.tsx b/app/src/organisms/RunProgressMeter/__tests__/InterventionTicks.test.tsx index 8c25d7359d9..c6562476818 100644 --- a/app/src/organisms/RunProgressMeter/__tests__/InterventionTicks.test.tsx +++ b/app/src/organisms/RunProgressMeter/__tests__/InterventionTicks.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/RunProgressMeter/__tests__/RunProgressMeter.test.tsx b/app/src/organisms/RunProgressMeter/__tests__/RunProgressMeter.test.tsx index 143a1f3afb4..2c517f601e6 100644 --- a/app/src/organisms/RunProgressMeter/__tests__/RunProgressMeter.test.tsx +++ b/app/src/organisms/RunProgressMeter/__tests__/RunProgressMeter.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { when } from 'vitest-when' import { screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' @@ -19,11 +19,12 @@ import { } from '../../InterventionModal' import { ProgressBar } from '/app/atoms/ProgressBar' import { useRunControls } from '../../RunTimeControl/hooks' -import { useMostRecentCompletedAnalysis } from '../../LabwarePositionCheck/useMostRecentCompletedAnalysis' import { useNotifyRunQuery, useNotifyAllCommandsQuery, useRunStatus, + useMostRecentCompletedAnalysis, + useLastRunCommand, } from '/app/resources/runs' import { useDownloadRunLog } from '../../Devices/hooks' import { @@ -34,7 +35,6 @@ import { import { RunProgressMeter } from '..' import { renderWithProviders } from '/app/__testing-utils__' -import { useLastRunCommand } from '../../Devices/hooks/useLastRunCommand' import { useRunningStepCounts } from '/app/resources/protocols/hooks' import type { RunCommandSummary } from '@opentrons/api-client' @@ -48,12 +48,10 @@ vi.mock('@opentrons/react-api-client', async importOriginal => { } }) vi.mock('../../RunTimeControl/hooks') -vi.mock('../../LabwarePositionCheck/useMostRecentCompletedAnalysis') vi.mock('/app/resources/runs') -vi.mock('../../Devices/hooks') vi.mock('/app/atoms/ProgressBar') vi.mock('../../InterventionModal') -vi.mock('../../Devices/hooks/useLastRunCommand') +vi.mock('../../Devices/hooks') vi.mock('/app/resources/protocols/hooks') vi.mock('/app/redux-resources/robots') diff --git a/app/src/organisms/RunProgressMeter/hooks/useRunProgressCopy.tsx b/app/src/organisms/RunProgressMeter/hooks/useRunProgressCopy.tsx index ee5cce24cea..66e1960f939 100644 --- a/app/src/organisms/RunProgressMeter/hooks/useRunProgressCopy.tsx +++ b/app/src/organisms/RunProgressMeter/hooks/useRunProgressCopy.tsx @@ -2,7 +2,7 @@ import { RUN_STATUS_BLOCKED_BY_OPEN_DOOR, RUN_STATUS_IDLE, } from '@opentrons/api-client' -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { getCommandTextData } from '/app/molecules/Command/utils/getCommandTextData' import { LegacyStyledText } from '@opentrons/components' diff --git a/app/src/organisms/RunProgressMeter/index.tsx b/app/src/organisms/RunProgressMeter/index.tsx index cd5ca1c3b24..078ea646687 100644 --- a/app/src/organisms/RunProgressMeter/index.tsx +++ b/app/src/organisms/RunProgressMeter/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import { css } from 'styled-components' @@ -29,7 +29,6 @@ import { RUN_STATUS_RUNNING, } from '@opentrons/api-client' -import { useMostRecentCompletedAnalysis } from '../LabwarePositionCheck/useMostRecentCompletedAnalysis' import { getModalPortalEl } from '../../App/portal' import { useRunControls } from '../RunTimeControl/hooks' import { InterventionModal, useInterventionModal } from '../InterventionModal' @@ -40,6 +39,7 @@ import { useNotifyRunQuery, useNotifyAllCommandsQuery, useRunStatus, + useMostRecentCompletedAnalysis, } from '/app/resources/runs' import { useRobotType } from '/app/redux-resources/robots' import { useRunningStepCounts } from '/app/resources/protocols/hooks' diff --git a/app/src/organisms/RunTimeControl/hooks.ts b/app/src/organisms/RunTimeControl/hooks.ts index 5e519baa2b0..d2ddfaf4744 100644 --- a/app/src/organisms/RunTimeControl/hooks.ts +++ b/app/src/organisms/RunTimeControl/hooks.ts @@ -6,8 +6,8 @@ import { useRunStatus, useCloneRun, DEFAULT_RUN_QUERY_REFETCH_INTERVAL, + useMostRecentCompletedAnalysis, } from '/app/resources/runs' -import { useMostRecentCompletedAnalysis } from '../LabwarePositionCheck/useMostRecentCompletedAnalysis' import type { UseQueryOptions } from 'react-query' import type { RunStatus, Run, RunData } from '@opentrons/api-client' diff --git a/app/src/organisms/SendProtocolToFlexSlideout/__tests__/SendProtocolToFlexSlideout.test.tsx b/app/src/organisms/SendProtocolToFlexSlideout/__tests__/SendProtocolToFlexSlideout.test.tsx index 2c7c9aebe1c..d9d4532af73 100644 --- a/app/src/organisms/SendProtocolToFlexSlideout/__tests__/SendProtocolToFlexSlideout.test.tsx +++ b/app/src/organisms/SendProtocolToFlexSlideout/__tests__/SendProtocolToFlexSlideout.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/SendProtocolToFlexSlideout/index.tsx b/app/src/organisms/SendProtocolToFlexSlideout/index.tsx index 969f080fb3a..176fa22e3c9 100644 --- a/app/src/organisms/SendProtocolToFlexSlideout/index.tsx +++ b/app/src/organisms/SendProtocolToFlexSlideout/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' @@ -52,7 +52,7 @@ export function SendProtocolToFlexSlideout( } = storedProtocolData const { t } = useTranslation(['protocol_details', 'protocol_list']) - const [selectedRobot, setSelectedRobot] = React.useState(null) + const [selectedRobot, setSelectedRobot] = useState(null) const isSelectedRobotOnDifferentSoftwareVersion = useIsRobotOnWrongVersionOfSoftware( selectedRobot?.name ?? '' diff --git a/app/src/organisms/TakeoverModal/TakeoverModal.tsx b/app/src/organisms/TakeoverModal/TakeoverModal.tsx index 713df38cdc7..b76d6e98d53 100644 --- a/app/src/organisms/TakeoverModal/TakeoverModal.tsx +++ b/app/src/organisms/TakeoverModal/TakeoverModal.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' diff --git a/app/src/organisms/TakeoverModal/__tests__/MaintenanceRunTakeover.test.tsx b/app/src/organisms/TakeoverModal/__tests__/MaintenanceRunTakeover.test.tsx index c67518c2ecc..94a59aa903a 100644 --- a/app/src/organisms/TakeoverModal/__tests__/MaintenanceRunTakeover.test.tsx +++ b/app/src/organisms/TakeoverModal/__tests__/MaintenanceRunTakeover.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/TakeoverModal/__tests__/TakeoverModal.test.tsx b/app/src/organisms/TakeoverModal/__tests__/TakeoverModal.test.tsx index 96314e1b6ce..a902544e4a0 100644 --- a/app/src/organisms/TakeoverModal/__tests__/TakeoverModal.test.tsx +++ b/app/src/organisms/TakeoverModal/__tests__/TakeoverModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/TakeoverModal/useMaintenanceRunTakeover.ts b/app/src/organisms/TakeoverModal/useMaintenanceRunTakeover.ts index 3c6373dee07..f166cc9a9d2 100644 --- a/app/src/organisms/TakeoverModal/useMaintenanceRunTakeover.ts +++ b/app/src/organisms/TakeoverModal/useMaintenanceRunTakeover.ts @@ -1,7 +1,7 @@ -import * as React from 'react' +import { useContext } from 'react' import { MaintenanceRunContext } from './MaintenanceRunStatusProvider' import type { MaintenanceRunStatus } from './MaintenanceRunStatusProvider' export function useMaintenanceRunTakeover(): MaintenanceRunStatus { - return React.useContext(MaintenanceRunContext) + return useContext(MaintenanceRunContext) } diff --git a/app/src/organisms/TaskList/TaskList.stories.tsx b/app/src/organisms/TaskList/TaskList.stories.tsx index b17bf6204d2..840ece49ed6 100644 --- a/app/src/organisms/TaskList/TaskList.stories.tsx +++ b/app/src/organisms/TaskList/TaskList.stories.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { TaskList as TaskListComponent } from './' import type { Story, Meta } from '@storybook/react' diff --git a/app/src/organisms/TaskList/index.tsx b/app/src/organisms/TaskList/index.tsx index 0fd37fc0394..2cebd9b1be2 100644 --- a/app/src/organisms/TaskList/index.tsx +++ b/app/src/organisms/TaskList/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect, Fragment } from 'react' import { ALIGN_CENTER, @@ -134,7 +134,7 @@ function ProgressTrackerItem({ const isFinalSubTaskOfTaskList = isLastSubTask && isLastTask return ( - + {/* subtask circle icon component */} - + ) })} @@ -342,11 +342,11 @@ function Task({ const hasSubTasks = subTasks.length > 0 const isDisabled = generalTaskDisabledReason != null - const [isTaskOpen, setIsTaskOpen] = React.useState( + const [isTaskOpen, setIsTaskOpen] = useState( hasSubTasks && isActiveTask ) - React.useEffect(() => { + useEffect(() => { setIsTaskOpen(hasSubTasks && isActiveTask) }, [isActiveTask, hasSubTasks]) diff --git a/app/src/organisms/ToasterOven/ToasterContext.ts b/app/src/organisms/ToasterOven/ToasterContext.ts index 0634a22ae5d..3467da62872 100644 --- a/app/src/organisms/ToasterOven/ToasterContext.ts +++ b/app/src/organisms/ToasterOven/ToasterContext.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { createContext } from 'react' import type { ToastProps, @@ -25,7 +25,7 @@ export interface ToasterContextType { makeSnackbar: MakeSnackbar } -export const ToasterContext = React.createContext({ +export const ToasterContext = createContext({ eatToast: () => {}, makeToast: () => '', makeSnackbar: () => {}, diff --git a/app/src/organisms/ToasterOven/hooks.ts b/app/src/organisms/ToasterOven/hooks.ts index 5c37a0e4cb0..555a08768bb 100644 --- a/app/src/organisms/ToasterOven/hooks.ts +++ b/app/src/organisms/ToasterOven/hooks.ts @@ -1,11 +1,11 @@ -import * as React from 'react' +import { useContext } from 'react' import { ToasterContext } from './ToasterContext' import type { ToasterContextType } from './ToasterContext' export function useToaster(): ToasterContextType { - const { eatToast, makeToast, makeSnackbar } = React.useContext(ToasterContext) + const { eatToast, makeToast, makeSnackbar } = useContext(ToasterContext) return { eatToast, makeToast, makeSnackbar } } diff --git a/app/src/organisms/UpdateAppModal/__tests__/UpdateAppModal.test.tsx b/app/src/organisms/UpdateAppModal/__tests__/UpdateAppModal.test.tsx index 83cd713c2ef..3131bee25a3 100644 --- a/app/src/organisms/UpdateAppModal/__tests__/UpdateAppModal.test.tsx +++ b/app/src/organisms/UpdateAppModal/__tests__/UpdateAppModal.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen, fireEvent } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/UpdateAppModal/index.tsx b/app/src/organisms/UpdateAppModal/index.tsx index b29c4e66528..450824c27f5 100644 --- a/app/src/organisms/UpdateAppModal/index.tsx +++ b/app/src/organisms/UpdateAppModal/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useSelector, useDispatch } from 'react-redux' import styled, { css } from 'styled-components' import { useNavigate } from 'react-router-dom' @@ -6,6 +5,7 @@ import { useTranslation } from 'react-i18next' import { ALIGN_CENTER, + Banner, BORDERS, COLORS, DIRECTION_COLUMN, @@ -28,7 +28,6 @@ import { import { ExternalLink } from '/app/atoms/Link/ExternalLink' import { ReleaseNotes } from '/app/molecules/ReleaseNotes' -import { Banner } from '/app/atoms/Banner' import { ProgressBar } from '/app/atoms/ProgressBar' import { useRemoveActiveAppUpdateToast } from '../Alerts' diff --git a/app/src/organisms/UpdateRobotBanner/__tests__/UpdateRobotBanner.test.tsx b/app/src/organisms/UpdateRobotBanner/__tests__/UpdateRobotBanner.test.tsx index 6742d808999..e7d90f70ea3 100644 --- a/app/src/organisms/UpdateRobotBanner/__tests__/UpdateRobotBanner.test.tsx +++ b/app/src/organisms/UpdateRobotBanner/__tests__/UpdateRobotBanner.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/organisms/UpdateRobotBanner/index.tsx b/app/src/organisms/UpdateRobotBanner/index.tsx index 57fcae39aff..e5d7d2d0e85 100644 --- a/app/src/organisms/UpdateRobotBanner/index.tsx +++ b/app/src/organisms/UpdateRobotBanner/index.tsx @@ -1,15 +1,15 @@ -import * as React from 'react' +import type * as React from 'react' import { useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' import { Btn, DIRECTION_COLUMN, Flex, + Banner, SPACING, LegacyStyledText, TYPOGRAPHY, } from '@opentrons/components' -import { Banner } from '/app/atoms/Banner' import { getRobotUpdateDisplayInfo } from '/app/redux/robot-update' import { handleUpdateBuildroot } from '../Devices/RobotSettings/UpdateBuildroot' diff --git a/app/src/organisms/UpdateRobotSoftware/CheckUpdates.tsx b/app/src/organisms/UpdateRobotSoftware/CheckUpdates.tsx index e23ff5417f6..9df29bc6280 100644 --- a/app/src/organisms/UpdateRobotSoftware/CheckUpdates.tsx +++ b/app/src/organisms/UpdateRobotSoftware/CheckUpdates.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/UpdateRobotSoftware/CompleteUpdateSoftware.tsx b/app/src/organisms/UpdateRobotSoftware/CompleteUpdateSoftware.tsx index f75a1777146..9f29911ac88 100644 --- a/app/src/organisms/UpdateRobotSoftware/CompleteUpdateSoftware.tsx +++ b/app/src/organisms/UpdateRobotSoftware/CompleteUpdateSoftware.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/UpdateRobotSoftware/ErrorUpdateSoftware.tsx b/app/src/organisms/UpdateRobotSoftware/ErrorUpdateSoftware.tsx index 22324c25b95..f1629e15b64 100644 --- a/app/src/organisms/UpdateRobotSoftware/ErrorUpdateSoftware.tsx +++ b/app/src/organisms/UpdateRobotSoftware/ErrorUpdateSoftware.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/UpdateRobotSoftware/NoUpdateFound.tsx b/app/src/organisms/UpdateRobotSoftware/NoUpdateFound.tsx index 35eed9fcc1b..27b466f06d9 100644 --- a/app/src/organisms/UpdateRobotSoftware/NoUpdateFound.tsx +++ b/app/src/organisms/UpdateRobotSoftware/NoUpdateFound.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/UpdateRobotSoftware/UpdateSoftware.tsx b/app/src/organisms/UpdateRobotSoftware/UpdateSoftware.tsx index 67c0cd1a3bc..9c499d8a4f3 100644 --- a/app/src/organisms/UpdateRobotSoftware/UpdateSoftware.tsx +++ b/app/src/organisms/UpdateRobotSoftware/UpdateSoftware.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { diff --git a/app/src/organisms/UpdateRobotSoftware/__tests__/CheckUpdates.test.tsx b/app/src/organisms/UpdateRobotSoftware/__tests__/CheckUpdates.test.tsx index 80bef11873f..d56c7305a0d 100644 --- a/app/src/organisms/UpdateRobotSoftware/__tests__/CheckUpdates.test.tsx +++ b/app/src/organisms/UpdateRobotSoftware/__tests__/CheckUpdates.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { describe, it } from 'vitest' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/organisms/UpdateRobotSoftware/__tests__/CompleteUpdateSoftware.test.tsx b/app/src/organisms/UpdateRobotSoftware/__tests__/CompleteUpdateSoftware.test.tsx index ab9f20849e1..b6b91424b92 100644 --- a/app/src/organisms/UpdateRobotSoftware/__tests__/CompleteUpdateSoftware.test.tsx +++ b/app/src/organisms/UpdateRobotSoftware/__tests__/CompleteUpdateSoftware.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/UpdateRobotSoftware/__tests__/ErrorUpdateSoftware.test.tsx b/app/src/organisms/UpdateRobotSoftware/__tests__/ErrorUpdateSoftware.test.tsx index eef021fe01e..d1706a4bf18 100644 --- a/app/src/organisms/UpdateRobotSoftware/__tests__/ErrorUpdateSoftware.test.tsx +++ b/app/src/organisms/UpdateRobotSoftware/__tests__/ErrorUpdateSoftware.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/UpdateRobotSoftware/__tests__/NoUpdateFound.test.tsx b/app/src/organisms/UpdateRobotSoftware/__tests__/NoUpdateFound.test.tsx index 5e6fec3bdca..2f276dc2f20 100644 --- a/app/src/organisms/UpdateRobotSoftware/__tests__/NoUpdateFound.test.tsx +++ b/app/src/organisms/UpdateRobotSoftware/__tests__/NoUpdateFound.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { describe, it, vi, expect } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/UpdateRobotSoftware/__tests__/UpdateRobotSoftware.test.tsx b/app/src/organisms/UpdateRobotSoftware/__tests__/UpdateRobotSoftware.test.tsx index 86bbb93b1ec..9df863fe5ce 100644 --- a/app/src/organisms/UpdateRobotSoftware/__tests__/UpdateRobotSoftware.test.tsx +++ b/app/src/organisms/UpdateRobotSoftware/__tests__/UpdateRobotSoftware.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, vi, beforeEach, expect } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/UpdateRobotSoftware/__tests__/UpdateSoftware.test.tsx b/app/src/organisms/UpdateRobotSoftware/__tests__/UpdateSoftware.test.tsx index 2a54550eae2..93deeb27956 100644 --- a/app/src/organisms/UpdateRobotSoftware/__tests__/UpdateSoftware.test.tsx +++ b/app/src/organisms/UpdateRobotSoftware/__tests__/UpdateSoftware.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { screen } from '@testing-library/react' import { describe, it, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' diff --git a/app/src/organisms/UpdateRobotSoftware/index.tsx b/app/src/organisms/UpdateRobotSoftware/index.tsx index 42b907ddb17..785494410d4 100644 --- a/app/src/organisms/UpdateRobotSoftware/index.tsx +++ b/app/src/organisms/UpdateRobotSoftware/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { useDispatch, useSelector } from 'react-redux' import type { Dispatch } from '/app/redux/types' @@ -41,9 +41,9 @@ export function UpdateRobotSoftware( step: null, error: null, } - const [isDownloading, setIsDownloading] = React.useState(false) + const [isDownloading, setIsDownloading] = useState(false) - React.useEffect(() => { + useEffect(() => { // check isDownloading to avoid dispatching again if (!isDownloading) { setIsDownloading(true) diff --git a/app/src/organisms/WellSelection/index.tsx b/app/src/organisms/WellSelection/index.tsx index 19cd9e04363..656745c0056 100644 --- a/app/src/organisms/WellSelection/index.tsx +++ b/app/src/organisms/WellSelection/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import reduce from 'lodash/reduce' import { COLORS, Labware, RobotCoordinateSpace } from '@opentrons/components' @@ -33,7 +33,7 @@ export function WellSelection(props: WellSelectionProps): JSX.Element { selectWells, channels, } = props - const [highlightedWells, setHighlightedWells] = React.useState({}) + const [highlightedWells, setHighlightedWells] = useState({}) const _wellsFromSelected: ( selectedWells: WellGroup diff --git a/app/src/pages/Desktop/AppSettings/AdvancedSettings.tsx b/app/src/pages/Desktop/AppSettings/AdvancedSettings.tsx index 7a0e0a6e032..f444c0e676f 100644 --- a/app/src/pages/Desktop/AppSettings/AdvancedSettings.tsx +++ b/app/src/pages/Desktop/AppSettings/AdvancedSettings.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { css } from 'styled-components' import { diff --git a/app/src/pages/Desktop/AppSettings/GeneralSettings.tsx b/app/src/pages/Desktop/AppSettings/GeneralSettings.tsx index 45918056041..96fd7ef0b35 100644 --- a/app/src/pages/Desktop/AppSettings/GeneralSettings.tsx +++ b/app/src/pages/Desktop/AppSettings/GeneralSettings.tsx @@ -1,5 +1,5 @@ // app info card with version and updated -import * as React from 'react' +import { useState } from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import { useSelector, useDispatch } from 'react-redux' @@ -7,6 +7,7 @@ import { useSelector, useDispatch } from 'react-redux' import { ALIGN_CENTER, ALIGN_START, + Banner, Box, COLORS, DIRECTION_COLUMN, @@ -24,7 +25,6 @@ import { import { TertiaryButton, ToggleButton } from '/app/atoms/buttons' import { ExternalLink } from '/app/atoms/Link/ExternalLink' import { Divider } from '/app/atoms/structure' -import { Banner } from '/app/atoms/Banner' import { CURRENT_VERSION, getAvailableShellUpdate, @@ -60,22 +60,21 @@ export function GeneralSettings(): JSX.Element { const [ showPreviousVersionModal, setShowPreviousVersionModal, - ] = React.useState(false) + ] = useState(false) const updateAvailable = Boolean(useSelector(getAvailableShellUpdate)) - const [showUpdateBanner, setShowUpdateBanner] = React.useState( + const [showUpdateBanner, setShowUpdateBanner] = useState( updateAvailable ) - const [ - showConnectRobotSlideout, - setShowConnectRobotSlideout, - ] = React.useState(false) + const [showConnectRobotSlideout, setShowConnectRobotSlideout] = useState( + false + ) // may be enabled, disabled, or unknown (because config is loading) const updateAlertEnabled = useSelector((s: State) => { const ignored = getAlertIsPermanentlyIgnored(s, ALERT_APP_UPDATE_AVAILABLE) return ignored !== null ? !ignored : null }) - const [showUpdateModal, setShowUpdateModal] = React.useState(false) + const [showUpdateModal, setShowUpdateModal] = useState(false) const handleToggle = (): void => { if (updateAlertEnabled !== null) { dispatch( diff --git a/app/src/pages/Desktop/AppSettings/PrivacySettings.tsx b/app/src/pages/Desktop/AppSettings/PrivacySettings.tsx index b0e75b3671c..abe45b61fa6 100644 --- a/app/src/pages/Desktop/AppSettings/PrivacySettings.tsx +++ b/app/src/pages/Desktop/AppSettings/PrivacySettings.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useSelector, useDispatch } from 'react-redux' diff --git a/app/src/pages/Desktop/AppSettings/__test__/AdvancedSettings.test.tsx b/app/src/pages/Desktop/AppSettings/__test__/AdvancedSettings.test.tsx index 2b5bcca9faf..0a1fcdb5ee4 100644 --- a/app/src/pages/Desktop/AppSettings/__test__/AdvancedSettings.test.tsx +++ b/app/src/pages/Desktop/AppSettings/__test__/AdvancedSettings.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { vi, it, describe, expect, beforeEach, afterEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/pages/Desktop/AppSettings/__test__/AppSettings.test.tsx b/app/src/pages/Desktop/AppSettings/__test__/AppSettings.test.tsx index 5002dec5407..cff6b6f93aa 100644 --- a/app/src/pages/Desktop/AppSettings/__test__/AppSettings.test.tsx +++ b/app/src/pages/Desktop/AppSettings/__test__/AppSettings.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, describe, beforeEach, it, expect, afterEach } from 'vitest' import { Route } from 'react-router' import { MemoryRouter, Routes } from 'react-router-dom' diff --git a/app/src/pages/Desktop/AppSettings/__test__/GeneralSettings.test.tsx b/app/src/pages/Desktop/AppSettings/__test__/GeneralSettings.test.tsx index 5965baac3e2..5cfd02e09a8 100644 --- a/app/src/pages/Desktop/AppSettings/__test__/GeneralSettings.test.tsx +++ b/app/src/pages/Desktop/AppSettings/__test__/GeneralSettings.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { vi, it, describe, expect, beforeEach, afterEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/pages/Desktop/AppSettings/__test__/PrivacySettings.test.tsx b/app/src/pages/Desktop/AppSettings/__test__/PrivacySettings.test.tsx index 9635ae518a7..3b0aaee80a4 100644 --- a/app/src/pages/Desktop/AppSettings/__test__/PrivacySettings.test.tsx +++ b/app/src/pages/Desktop/AppSettings/__test__/PrivacySettings.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, it, describe } from 'vitest' import { MemoryRouter } from 'react-router-dom' diff --git a/app/src/pages/Desktop/AppSettings/index.tsx b/app/src/pages/Desktop/AppSettings/index.tsx index f11aa930187..00b8b5ca0a0 100644 --- a/app/src/pages/Desktop/AppSettings/index.tsx +++ b/app/src/pages/Desktop/AppSettings/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' import { Navigate, useParams } from 'react-router-dom' diff --git a/app/src/pages/Desktop/Devices/CalibrationDashboard/__tests__/CalibrationDashboard.test.tsx b/app/src/pages/Desktop/Devices/CalibrationDashboard/__tests__/CalibrationDashboard.test.tsx index c8e4935bf21..f11f674ec3d 100644 --- a/app/src/pages/Desktop/Devices/CalibrationDashboard/__tests__/CalibrationDashboard.test.tsx +++ b/app/src/pages/Desktop/Devices/CalibrationDashboard/__tests__/CalibrationDashboard.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, describe, it, beforeEach } from 'vitest' import { screen } from '@testing-library/react' import { MemoryRouter, Route, Routes } from 'react-router-dom' @@ -7,16 +6,14 @@ import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' import { CalibrationDashboard } from '..' -import { - useCalibrationTaskList, - useAttachedPipettes, -} from '/app/organisms/Devices/hooks' +import { useCalibrationTaskList } from '/app/organisms/Devices/hooks' import { useDashboardCalibratePipOffset } from '../hooks/useDashboardCalibratePipOffset' import { useDashboardCalibrateTipLength } from '../hooks/useDashboardCalibrateTipLength' import { useDashboardCalibrateDeck } from '../hooks/useDashboardCalibrateDeck' import { expectedTaskList } from '/app/organisms/Devices/hooks/__fixtures__/taskListFixtures' import { mockLeftProtoPipette } from '/app/redux/pipettes/__fixtures__' import { useNotifyAllRunsQuery } from '/app/resources/runs' +import { useAttachedPipettes } from '/app/resources/instruments' vi.mock('/app/redux-resources/robots') vi.mock('/app/organisms/Devices/hooks') @@ -24,6 +21,7 @@ vi.mock('../hooks/useDashboardCalibratePipOffset') vi.mock('../hooks/useDashboardCalibrateTipLength') vi.mock('../hooks/useDashboardCalibrateDeck') vi.mock('/app/resources/runs') +vi.mock('/app/resources/instruments') const render = (path = '/') => { return renderWithProviders( diff --git a/app/src/pages/Desktop/Devices/CalibrationDashboard/hooks/useDashboardCalibrateDeck.tsx b/app/src/pages/Desktop/Devices/CalibrationDashboard/hooks/useDashboardCalibrateDeck.tsx index 9ce182dd812..908ad25aa3a 100644 --- a/app/src/pages/Desktop/Devices/CalibrationDashboard/hooks/useDashboardCalibrateDeck.tsx +++ b/app/src/pages/Desktop/Devices/CalibrationDashboard/hooks/useDashboardCalibrateDeck.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useRef } from 'react' import { createPortal } from 'react-dom' import { useSelector } from 'react-redux' import { useTranslation } from 'react-i18next' @@ -26,11 +26,11 @@ const spinnerCommandBlockList: SessionCommandString[] = [ export function useDashboardCalibrateDeck( robotName: string ): [DashboardCalDeckInvoker, JSX.Element | null, boolean] { - const trackedRequestId = React.useRef(null) - const createRequestId = React.useRef(null) - const jogRequestId = React.useRef(null) - const exitBeforeDeckConfigCompletion = React.useRef(false) - const invalidateHandlerRef = React.useRef<(() => void) | undefined>() + const trackedRequestId = useRef(null) + const createRequestId = useRef(null) + const jogRequestId = useRef(null) + const exitBeforeDeckConfigCompletion = useRef(false) + const invalidateHandlerRef = useRef<(() => void) | undefined>() const { t } = useTranslation('robot_calibration') const deckCalSession: DeckCalibrationSession | null = useSelector( diff --git a/app/src/pages/Desktop/Devices/CalibrationDashboard/hooks/useDashboardCalibratePipOffset.tsx b/app/src/pages/Desktop/Devices/CalibrationDashboard/hooks/useDashboardCalibratePipOffset.tsx index b78a616f6cc..12fef601e17 100644 --- a/app/src/pages/Desktop/Devices/CalibrationDashboard/hooks/useDashboardCalibratePipOffset.tsx +++ b/app/src/pages/Desktop/Devices/CalibrationDashboard/hooks/useDashboardCalibratePipOffset.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useRef, useEffect } from 'react' import { createPortal } from 'react-dom' import { useSelector, useDispatch } from 'react-redux' import { useTranslation } from 'react-i18next' @@ -29,10 +29,10 @@ export function useDashboardCalibratePipOffset( robotName: string, onComplete: (() => unknown) | null = null ): [DashboardCalOffsetInvoker, JSX.Element | null] { - const createRequestId = React.useRef(null) - const deleteRequestId = React.useRef(null) - const jogRequestId = React.useRef(null) - const spinnerRequestId = React.useRef(null) + const createRequestId = useRef(null) + const deleteRequestId = useRef(null) + const jogRequestId = useRef(null) + const spinnerRequestId = useRef(null) const dispatch = useDispatch() const { t } = useTranslation('robot_calibration') @@ -112,7 +112,7 @@ export function useDashboardCalibratePipOffset( : null )?.status === RobotApi.PENDING - React.useEffect(() => { + useEffect(() => { if (shouldClose) { onComplete?.() deleteRequestId.current = null diff --git a/app/src/pages/Desktop/Devices/CalibrationDashboard/hooks/useDashboardCalibrateTipLength.tsx b/app/src/pages/Desktop/Devices/CalibrationDashboard/hooks/useDashboardCalibrateTipLength.tsx index d8da4848f5f..e26e485801a 100644 --- a/app/src/pages/Desktop/Devices/CalibrationDashboard/hooks/useDashboardCalibrateTipLength.tsx +++ b/app/src/pages/Desktop/Devices/CalibrationDashboard/hooks/useDashboardCalibrateTipLength.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useRef, useState } from 'react' import { createPortal } from 'react-dom' import { useSelector, useDispatch } from 'react-redux' import { useTranslation } from 'react-i18next' @@ -32,15 +32,15 @@ const spinnerCommandBlockList: SessionCommandString[] = [ export function useDashboardCalibrateTipLength( robotName: string ): [DashboardCalTipLengthInvoker, JSX.Element | null] { - const createRequestId = React.useRef(null) - const trackedRequestId = React.useRef(null) - const jogRequestId = React.useRef(null) - const sessionParams = React.useRef< + const createRequestId = useRef(null) + const trackedRequestId = useRef(null) + const jogRequestId = useRef(null) + const sessionParams = useRef< | (Pick & Partial>) | null >(null) - const invalidateHandlerRef = React.useRef<(() => void) | undefined>() + const invalidateHandlerRef = useRef<(() => void) | undefined>() const dispatch = useDispatch() const { t } = useTranslation('robot_calibration') @@ -86,9 +86,9 @@ export function useDashboardCalibrateTipLength( ) const configHasCalibrationBlock = useSelector(getHasCalibrationBlock) - const [showCalBlockModal, setShowCalBlockModal] = React.useState< - boolean | null - >(null) + const [showCalBlockModal, setShowCalBlockModal] = useState( + null + ) const handleStartDashboardTipLengthCalSession: DashboardCalTipLengthInvoker = props => { const { params, hasBlockModalResponse, invalidateHandler } = props diff --git a/app/src/pages/Desktop/Devices/CalibrationDashboard/index.tsx b/app/src/pages/Desktop/Devices/CalibrationDashboard/index.tsx index 2a29daa1fd4..512dbdfe35d 100644 --- a/app/src/pages/Desktop/Devices/CalibrationDashboard/index.tsx +++ b/app/src/pages/Desktop/Devices/CalibrationDashboard/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useParams } from 'react-router-dom' import { ApiHostProvider } from '@opentrons/react-api-client' import { CalibrationTaskList } from '/app/organisms/CalibrationTaskList' diff --git a/app/src/pages/Desktop/Devices/DeviceDetails/DeviceDetailsComponent.tsx b/app/src/pages/Desktop/Devices/DeviceDetails/DeviceDetailsComponent.tsx index 63ed9ca0991..4812edcb0f5 100644 --- a/app/src/pages/Desktop/Devices/DeviceDetails/DeviceDetailsComponent.tsx +++ b/app/src/pages/Desktop/Devices/DeviceDetails/DeviceDetailsComponent.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useEstopQuery } from '@opentrons/react-api-client' import { ALIGN_CENTER, diff --git a/app/src/pages/Desktop/Devices/DeviceDetails/__tests__/DeviceDetails.test.tsx b/app/src/pages/Desktop/Devices/DeviceDetails/__tests__/DeviceDetails.test.tsx index 4729c3d6b59..ff3963f95f3 100644 --- a/app/src/pages/Desktop/Devices/DeviceDetails/__tests__/DeviceDetails.test.tsx +++ b/app/src/pages/Desktop/Devices/DeviceDetails/__tests__/DeviceDetails.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, it, describe, expect, beforeEach } from 'vitest' import { when } from 'vitest-when' import { screen } from '@testing-library/react' diff --git a/app/src/pages/Desktop/Devices/DeviceDetails/__tests__/DeviceDetailsComponent.test.tsx b/app/src/pages/Desktop/Devices/DeviceDetails/__tests__/DeviceDetailsComponent.test.tsx index a436cbe672a..418f638464d 100644 --- a/app/src/pages/Desktop/Devices/DeviceDetails/__tests__/DeviceDetailsComponent.test.tsx +++ b/app/src/pages/Desktop/Devices/DeviceDetails/__tests__/DeviceDetailsComponent.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, it, describe, expect, beforeEach } from 'vitest' import { when } from 'vitest-when' diff --git a/app/src/pages/Desktop/Devices/DeviceDetails/index.tsx b/app/src/pages/Desktop/Devices/DeviceDetails/index.tsx index 1a261c43b41..b96ffdb3b45 100644 --- a/app/src/pages/Desktop/Devices/DeviceDetails/index.tsx +++ b/app/src/pages/Desktop/Devices/DeviceDetails/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useSelector } from 'react-redux' import { Navigate, useParams } from 'react-router-dom' diff --git a/app/src/pages/Desktop/Devices/DevicesLanding/NewRobotSetupHelp.tsx b/app/src/pages/Desktop/Devices/DevicesLanding/NewRobotSetupHelp.tsx index c9c9ea6d2cd..80747a7607b 100644 --- a/app/src/pages/Desktop/Devices/DevicesLanding/NewRobotSetupHelp.tsx +++ b/app/src/pages/Desktop/Devices/DevicesLanding/NewRobotSetupHelp.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { createPortal } from 'react-dom' import { useTranslation } from 'react-i18next' import { @@ -24,9 +24,7 @@ const NEW_OT2_SETUP_SUPPORT_ARTICLE_HREF = export function NewRobotSetupHelp(): JSX.Element { const { t } = useTranslation(['devices_landing', 'shared', 'branded']) - const [showNewRobotHelpModal, setShowNewRobotHelpModal] = React.useState( - false - ) + const [showNewRobotHelpModal, setShowNewRobotHelpModal] = useState(false) return ( <> diff --git a/app/src/pages/Desktop/Devices/DevicesLanding/__tests__/DevicesLanding.test.tsx b/app/src/pages/Desktop/Devices/DevicesLanding/__tests__/DevicesLanding.test.tsx index b23625de551..b2548ac50ac 100644 --- a/app/src/pages/Desktop/Devices/DevicesLanding/__tests__/DevicesLanding.test.tsx +++ b/app/src/pages/Desktop/Devices/DevicesLanding/__tests__/DevicesLanding.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect, beforeEach, afterEach } from 'vitest' diff --git a/app/src/pages/Desktop/Devices/DevicesLanding/__tests__/NewRobotSetupHelp.test.tsx b/app/src/pages/Desktop/Devices/DevicesLanding/__tests__/NewRobotSetupHelp.test.tsx index 7c21a22a938..cc90f46b41d 100644 --- a/app/src/pages/Desktop/Devices/DevicesLanding/__tests__/NewRobotSetupHelp.test.tsx +++ b/app/src/pages/Desktop/Devices/DevicesLanding/__tests__/NewRobotSetupHelp.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { it, describe, expect } from 'vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/pages/Desktop/Devices/DevicesLanding/index.tsx b/app/src/pages/Desktop/Devices/DevicesLanding/index.tsx index 7ee8afd2993..7e53817e3ec 100644 --- a/app/src/pages/Desktop/Devices/DevicesLanding/index.tsx +++ b/app/src/pages/Desktop/Devices/DevicesLanding/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useSelector } from 'react-redux' import partition from 'lodash/partition' diff --git a/app/src/pages/Desktop/Devices/ProtocolRunDetails/__tests__/ProtocolRunDetails.test.tsx b/app/src/pages/Desktop/Devices/ProtocolRunDetails/__tests__/ProtocolRunDetails.test.tsx index 8c6e304cf42..192c2e6971d 100644 --- a/app/src/pages/Desktop/Devices/ProtocolRunDetails/__tests__/ProtocolRunDetails.test.tsx +++ b/app/src/pages/Desktop/Devices/ProtocolRunDetails/__tests__/ProtocolRunDetails.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, it, describe, expect, beforeEach, afterEach } from 'vitest' import { Route, MemoryRouter, Routes } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' @@ -7,26 +6,24 @@ import { when } from 'vitest-when' import { renderWithProviders } from '/app/__testing-utils__' import { i18n } from '/app/i18n' import { mockConnectableRobot } from '/app/redux/discovery/__fixtures__' -import { - useModuleRenderInfoForProtocolById, - useRunStatuses, - useSyncRobotClock, - useRunHasStarted, -} from '/app/organisms/Devices/hooks' -import { useMostRecentCompletedAnalysis } from '/app/organisms/LabwarePositionCheck/useMostRecentCompletedAnalysis' +import { useRunStatuses, useSyncRobotClock } from '/app/organisms/Devices/hooks' import { ProtocolRunHeader } from '/app/organisms/Devices/ProtocolRun/ProtocolRunHeader' import { ProtocolRunModuleControls } from '/app/organisms/Devices/ProtocolRun/ProtocolRunModuleControls' import { ProtocolRunSetup } from '/app/organisms/Devices/ProtocolRun/ProtocolRunSetup' import { RunPreviewComponent } from '/app/organisms/RunPreview' import { ProtocolRunRuntimeParameters } from '/app/organisms/Devices/ProtocolRun/ProtocolRunRunTimeParameters' -import { useCurrentRunId } from '/app/resources/runs' +import { + useCurrentRunId, + useMostRecentCompletedAnalysis, + useRunHasStarted, + useModuleRenderInfoForProtocolById, +} from '/app/resources/runs' import { mockRobotSideAnalysis } from '/app/molecules/Command/__fixtures__' import { useRobot } from '/app/redux-resources/robots' import { ProtocolRunDetails } from '..' import type { ModuleModel, ModuleType } from '@opentrons/shared-data' -vi.mock('/app/organisms/LabwarePositionCheck/useMostRecentCompletedAnalysis') vi.mock('/app/organisms/Devices/hooks') vi.mock('/app/organisms/Devices/ProtocolRun/ProtocolRunHeader') vi.mock('/app/organisms/Devices/ProtocolRun/ProtocolRunSetup') diff --git a/app/src/pages/Desktop/Devices/ProtocolRunDetails/index.tsx b/app/src/pages/Desktop/Devices/ProtocolRunDetails/index.tsx index ee7d82553d4..53bc8d60c3a 100644 --- a/app/src/pages/Desktop/Devices/ProtocolRunDetails/index.tsx +++ b/app/src/pages/Desktop/Devices/ProtocolRunDetails/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useEffect, useRef, useState } from 'react' import isEmpty from 'lodash/isEmpty' import { useTranslation } from 'react-i18next' import { useDispatch } from 'react-redux' @@ -22,12 +22,7 @@ import { useHoverTooltip, } from '@opentrons/components' import { ApiHostProvider } from '@opentrons/react-api-client' -import { - useModuleRenderInfoForProtocolById, - useRunHasStarted, - useRunStatuses, - useSyncRobotClock, -} from '/app/organisms/Devices/hooks' +import { useRunStatuses, useSyncRobotClock } from '/app/organisms/Devices/hooks' import { ProtocolRunHeader } from '/app/organisms/Devices/ProtocolRun/ProtocolRunHeader' import { RunPreview } from '/app/organisms/RunPreview' import { @@ -37,11 +32,15 @@ import { import { BackToTopButton } from '/app/organisms/Devices/ProtocolRun/BackToTopButton' import { ProtocolRunModuleControls } from '/app/organisms/Devices/ProtocolRun/ProtocolRunModuleControls' import { ProtocolRunRuntimeParameters } from '/app/organisms/Devices/ProtocolRun/ProtocolRunRunTimeParameters' -import { useCurrentRunId } from '/app/resources/runs' +import { + useCurrentRunId, + useMostRecentCompletedAnalysis, + useRunHasStarted, + useModuleRenderInfoForProtocolById, +} from '/app/resources/runs' import { OPENTRONS_USB } from '/app/redux/discovery' import { fetchProtocols } from '/app/redux/protocol-storage' import { appShellRequestor } from '/app/redux/shell/remote' -import { useMostRecentCompletedAnalysis } from '/app/organisms/LabwarePositionCheck/useMostRecentCompletedAnalysis' import { useRobot, useRobotType } from '/app/redux-resources/robots' import type { ViewportListRef } from 'react-viewport-list' @@ -137,7 +136,7 @@ export function ProtocolRunDetails(): JSX.Element | null { const robot = useRobot(robotName) useSyncRobotClock(robotName) - React.useEffect(() => { + useEffect(() => { dispatch(fetchProtocols()) }, [dispatch]) return robot != null ? ( @@ -178,11 +177,11 @@ interface PageContentsProps { function PageContents(props: PageContentsProps): JSX.Element { const { runId, robotName, protocolRunDetailsTab } = props const robotType = useRobotType(robotName) - const protocolRunHeaderRef = React.useRef(null) - const listRef = React.useRef(null) - const [jumpedIndex, setJumpedIndex] = React.useState(null) + const protocolRunHeaderRef = useRef(null) + const listRef = useRef(null) + const [jumpedIndex, setJumpedIndex] = useState(null) - React.useEffect(() => { + useEffect(() => { if (jumpedIndex != null) { setTimeout(() => { setJumpedIndex(null) @@ -190,7 +189,7 @@ function PageContents(props: PageContentsProps): JSX.Element { } }, [jumpedIndex]) - const [missingSteps, setMissingSteps] = React.useState< + const [missingSteps, setMissingSteps] = useState< ReturnType >(initialMissingSteps()) @@ -323,7 +322,7 @@ const SetupTab = (props: SetupTabProps): JSX.Element | null => { // On the initial render or when a run first begins, navigate to "run preview" if the run has started. // If "run again" is clicked, the user should NOT be directed back to the "setup" tab. - React.useEffect(() => { + useEffect(() => { if (runHasStarted && protocolRunDetailsTab !== 'run-preview') { navigate(`/devices/${robotName}/protocol-runs/${runId}/run-preview`) } @@ -352,7 +351,7 @@ const ParametersTab = (props: ParametersTabProps): JSX.Element | null => { const navigate = useNavigate() const disabled = mostRecentAnalysis == null - React.useEffect(() => { + useEffect(() => { if (disabled && protocolRunDetailsTab === 'runtime-parameters') { navigate(`/devices/${robotName}/protocol-runs/${runId}/run-preview`, { replace: true, @@ -394,7 +393,7 @@ const ModuleControlsTab = ( : 'not_available_for_a_run_in_progress' )}` - React.useEffect(() => { + useEffect(() => { if (disabled && protocolRunDetailsTab === 'module-controls') navigate(`/devices/${robotName}/protocol-runs/${runId}/run-preview`) }, [disabled, navigate, protocolRunDetailsTab, robotName, runId]) diff --git a/app/src/pages/Desktop/Devices/RobotSettings/__tests__/RobotSettings.test.tsx b/app/src/pages/Desktop/Devices/RobotSettings/__tests__/RobotSettings.test.tsx index aa8343c1bd8..7e0ed778aca 100644 --- a/app/src/pages/Desktop/Devices/RobotSettings/__tests__/RobotSettings.test.tsx +++ b/app/src/pages/Desktop/Devices/RobotSettings/__tests__/RobotSettings.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, it, describe, expect, beforeEach, afterEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' import { Route, MemoryRouter, Routes } from 'react-router-dom' diff --git a/app/src/pages/Desktop/Devices/RobotSettings/index.tsx b/app/src/pages/Desktop/Devices/RobotSettings/index.tsx index 15f289764b4..ffabfc447ea 100644 --- a/app/src/pages/Desktop/Devices/RobotSettings/index.tsx +++ b/app/src/pages/Desktop/Devices/RobotSettings/index.tsx @@ -1,8 +1,9 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { useParams, Navigate } from 'react-router-dom' import { + Banner, BORDERS, Box, COLORS, @@ -25,7 +26,6 @@ import { import { appShellRequestor } from '/app/redux/shell/remote' import { getRobotUpdateSession } from '/app/redux/robot-update' import { getDevtoolsEnabled } from '/app/redux/config' -import { Banner } from '/app/atoms/Banner' import { useRobot } from '/app/redux-resources/robots' import { Line } from '/app/atoms/structure' import { NavTab } from '/app/molecules/NavTab' @@ -48,9 +48,7 @@ export function RobotSettings(): JSX.Element | null { const robot = useRobot(robotName) const isCalibrationDisabled = robot?.status !== CONNECTABLE const isNetworkingDisabled = robot?.status === UNREACHABLE - const [showRobotBusyBanner, setShowRobotBusyBanner] = React.useState( - false - ) + const [showRobotBusyBanner, setShowRobotBusyBanner] = useState(false) const robotUpdateSession = useSelector(getRobotUpdateSession) const updateRobotStatus = (isRobotBusy: boolean): void => { diff --git a/app/src/pages/Desktop/Labware/__tests__/Labware.test.tsx b/app/src/pages/Desktop/Labware/__tests__/Labware.test.tsx index b5133473704..3c919b7f295 100644 --- a/app/src/pages/Desktop/Labware/__tests__/Labware.test.tsx +++ b/app/src/pages/Desktop/Labware/__tests__/Labware.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, it, describe, expect, beforeEach, afterEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' import { MemoryRouter } from 'react-router-dom' diff --git a/app/src/pages/Desktop/Labware/__tests__/hooks.test.tsx b/app/src/pages/Desktop/Labware/__tests__/hooks.test.tsx index 14783cd6fe5..5fe55c260dc 100644 --- a/app/src/pages/Desktop/Labware/__tests__/hooks.test.tsx +++ b/app/src/pages/Desktop/Labware/__tests__/hooks.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { Provider } from 'react-redux' import { createStore } from 'redux' import { vi, it, describe, expect, beforeEach, afterEach } from 'vitest' diff --git a/app/src/pages/Desktop/Labware/index.tsx b/app/src/pages/Desktop/Labware/index.tsx index 955f8a26fc1..159e57c306e 100644 --- a/app/src/pages/Desktop/Labware/index.tsx +++ b/app/src/pages/Desktop/Labware/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState, useEffect } from 'react' import { useTranslation } from 'react-i18next' import startCase from 'lodash/startCase' import { css } from 'styled-components' @@ -84,35 +84,33 @@ const SORT_BY_BUTTON_STYLE = css` export function Labware(): JSX.Element { const { t } = useTranslation(['labware_landing', 'shared']) const enableLabwareCreator = useFeatureFlag('enableLabwareCreator') - const [sortBy, setSortBy] = React.useState('alphabetical') - const [showSortByMenu, setShowSortByMenu] = React.useState(false) + const [sortBy, setSortBy] = useState('alphabetical') + const [showSortByMenu, setShowSortByMenu] = useState(false) const toggleSetShowSortByMenu = (): void => { setShowSortByMenu(!showSortByMenu) } const dispatch = useDispatch() - const [showLC, setShowLC] = React.useState(false) + const [showLC, setShowLC] = useState(false) const trackEvent = useTrackEvent() - const [filterBy, setFilterBy] = React.useState('all') + const [filterBy, setFilterBy] = useState('all') const { makeToast } = useToaster() const labware = useAllLabware(sortBy, filterBy) const { labwareFailureMessage, clearLabwareFailure } = useLabwareFailure() const { newLabwareName, clearLabwareName } = useNewLabwareName() - const [showAddLabwareSlideout, setShowAddLabwareSlideout] = React.useState( - false - ) + const [showAddLabwareSlideout, setShowAddLabwareSlideout] = useState(false) const [ currentLabwareDef, setCurrentLabwareDef, - ] = React.useState(null) + ] = useState(null) const sortOverflowWrapperRef = useOnClickOutside({ onClickOutside: () => { setShowSortByMenu(false) }, }) - React.useEffect(() => { + useEffect(() => { if (labwareFailureMessage != null) { setShowAddLabwareSlideout(false) makeToast(labwareFailureMessage, ERROR_TOAST, { diff --git a/app/src/pages/Desktop/Protocols/ProtocolDetails/ProtocolTimeline.tsx b/app/src/pages/Desktop/Protocols/ProtocolDetails/ProtocolTimeline.tsx index affaf82f9f3..f113d863888 100644 --- a/app/src/pages/Desktop/Protocols/ProtocolDetails/ProtocolTimeline.tsx +++ b/app/src/pages/Desktop/Protocols/ProtocolDetails/ProtocolTimeline.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useEffect } from 'react' import { useParams } from 'react-router-dom' import { useDispatch, useSelector } from 'react-redux' import { Icon, Box, SPACING } from '@opentrons/components' @@ -17,7 +17,7 @@ export function ProtocolTimeline(): JSX.Element { getStoredProtocol(state, protocolKey) ) - React.useEffect(() => { + useEffect(() => { dispatch(fetchProtocols()) }, []) diff --git a/app/src/pages/Desktop/Protocols/ProtocolDetails/__tests__/ProtocolDetails.test.tsx b/app/src/pages/Desktop/Protocols/ProtocolDetails/__tests__/ProtocolDetails.test.tsx index e51038afbc9..c710350180a 100644 --- a/app/src/pages/Desktop/Protocols/ProtocolDetails/__tests__/ProtocolDetails.test.tsx +++ b/app/src/pages/Desktop/Protocols/ProtocolDetails/__tests__/ProtocolDetails.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, it, describe, expect, beforeEach, afterEach } from 'vitest' import { screen } from '@testing-library/react' import { Route, MemoryRouter, Routes } from 'react-router-dom' diff --git a/app/src/pages/Desktop/Protocols/ProtocolDetails/index.tsx b/app/src/pages/Desktop/Protocols/ProtocolDetails/index.tsx index e147fab13b5..d2e97de4147 100644 --- a/app/src/pages/Desktop/Protocols/ProtocolDetails/index.tsx +++ b/app/src/pages/Desktop/Protocols/ProtocolDetails/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useEffect } from 'react' import { useParams, Navigate } from 'react-router-dom' import { useDispatch, useSelector } from 'react-redux' @@ -18,7 +18,7 @@ export function ProtocolDetails(): JSX.Element { getStoredProtocol(state, protocolKey) ) - React.useEffect(() => { + useEffect(() => { dispatch(fetchProtocols()) }, [dispatch]) diff --git a/app/src/pages/Desktop/Protocols/ProtocolsLanding/__tests__/ProtocolsLanding.test.tsx b/app/src/pages/Desktop/Protocols/ProtocolsLanding/__tests__/ProtocolsLanding.test.tsx index 61a35c9aefc..3b00caf21dd 100644 --- a/app/src/pages/Desktop/Protocols/ProtocolsLanding/__tests__/ProtocolsLanding.test.tsx +++ b/app/src/pages/Desktop/Protocols/ProtocolsLanding/__tests__/ProtocolsLanding.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, it, describe } from 'vitest' import { screen } from '@testing-library/react' import { renderWithProviders } from '/app/__testing-utils__' diff --git a/app/src/pages/Desktop/Protocols/ProtocolsLanding/index.tsx b/app/src/pages/Desktop/Protocols/ProtocolsLanding/index.tsx index 628594247c6..e600441fc00 100644 --- a/app/src/pages/Desktop/Protocols/ProtocolsLanding/index.tsx +++ b/app/src/pages/Desktop/Protocols/ProtocolsLanding/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useEffect } from 'react' import { useDispatch, useSelector } from 'react-redux' import { fetchProtocols, getStoredProtocols } from '/app/redux/protocol-storage' import { ProtocolsEmptyState } from '/app/organisms/ProtocolsLanding/ProtocolsEmptyState' @@ -11,7 +11,7 @@ export function ProtocolsLanding(): JSX.Element { const storedProtocols = useSelector((state: State) => getStoredProtocols(state) ) - React.useEffect(() => { + useEffect(() => { dispatch(fetchProtocols()) }, [dispatch]) diff --git a/app/src/pages/ODD/ConnectViaEthernet/DisplayConnectionStatus.tsx b/app/src/pages/ODD/ConnectViaEthernet/DisplayConnectionStatus.tsx index 90bd374b948..746b04f2e24 100644 --- a/app/src/pages/ODD/ConnectViaEthernet/DisplayConnectionStatus.tsx +++ b/app/src/pages/ODD/ConnectViaEthernet/DisplayConnectionStatus.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useNavigate } from 'react-router-dom' diff --git a/app/src/pages/ODD/ConnectViaEthernet/TitleHeader.tsx b/app/src/pages/ODD/ConnectViaEthernet/TitleHeader.tsx index c837499c22b..30715fdab2f 100644 --- a/app/src/pages/ODD/ConnectViaEthernet/TitleHeader.tsx +++ b/app/src/pages/ODD/ConnectViaEthernet/TitleHeader.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useNavigate } from 'react-router-dom' import { diff --git a/app/src/pages/ODD/ConnectViaEthernet/__tests__/ConnectViaEthernet.test.tsx b/app/src/pages/ODD/ConnectViaEthernet/__tests__/ConnectViaEthernet.test.tsx index c3288101571..c785d8c5e88 100644 --- a/app/src/pages/ODD/ConnectViaEthernet/__tests__/ConnectViaEthernet.test.tsx +++ b/app/src/pages/ODD/ConnectViaEthernet/__tests__/ConnectViaEthernet.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { vi, it, describe, beforeEach, afterEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/pages/ODD/ConnectViaEthernet/__tests__/DisplayConnectionStatus.test.tsx b/app/src/pages/ODD/ConnectViaEthernet/__tests__/DisplayConnectionStatus.test.tsx index 17bbb8e85b1..9efbfd5c3dc 100644 --- a/app/src/pages/ODD/ConnectViaEthernet/__tests__/DisplayConnectionStatus.test.tsx +++ b/app/src/pages/ODD/ConnectViaEthernet/__tests__/DisplayConnectionStatus.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, describe, expect, beforeEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/pages/ODD/ConnectViaEthernet/__tests__/TitleHeader.test.tsx b/app/src/pages/ODD/ConnectViaEthernet/__tests__/TitleHeader.test.tsx index 8b40ae71671..cffa8e9c63a 100644 --- a/app/src/pages/ODD/ConnectViaEthernet/__tests__/TitleHeader.test.tsx +++ b/app/src/pages/ODD/ConnectViaEthernet/__tests__/TitleHeader.test.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { vi, it, describe, expect, beforeEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/pages/ODD/ConnectViaEthernet/index.tsx b/app/src/pages/ODD/ConnectViaEthernet/index.tsx index fe141f853ae..644d0616e1e 100644 --- a/app/src/pages/ODD/ConnectViaEthernet/index.tsx +++ b/app/src/pages/ODD/ConnectViaEthernet/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useTranslation } from 'react-i18next' import { useSelector, useDispatch } from 'react-redux' @@ -28,7 +28,7 @@ export function ConnectViaEthernet(): JSX.Element { const [ showNetworkDetailsModal, setShowNetworkDetailsModal, - ] = React.useState(false) + ] = useState(false) const { ethernet } = useSelector((state: State) => getNetworkInterfaces(state, robotName) diff --git a/app/src/pages/ODD/ConnectViaUSB/_tests__/ConnectedViaUSB.test.tsx b/app/src/pages/ODD/ConnectViaUSB/_tests__/ConnectedViaUSB.test.tsx index dea215d9026..4072cd1a7a3 100644 --- a/app/src/pages/ODD/ConnectViaUSB/_tests__/ConnectedViaUSB.test.tsx +++ b/app/src/pages/ODD/ConnectViaUSB/_tests__/ConnectedViaUSB.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, it, describe, expect, beforeEach, afterEach } from 'vitest' import { MemoryRouter } from 'react-router-dom' import { fireEvent } from '@testing-library/react' diff --git a/app/src/pages/ODD/ConnectViaUSB/index.tsx b/app/src/pages/ODD/ConnectViaUSB/index.tsx index e1ff45917e0..40ae95e63cf 100644 --- a/app/src/pages/ODD/ConnectViaUSB/index.tsx +++ b/app/src/pages/ODD/ConnectViaUSB/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useNavigate } from 'react-router-dom' import { diff --git a/app/src/pages/ODD/ConnectViaWifi/SelectAuthenticationType.tsx b/app/src/pages/ODD/ConnectViaWifi/SelectAuthenticationType.tsx index 9d367353d53..ad033d3ba7e 100644 --- a/app/src/pages/ODD/ConnectViaWifi/SelectAuthenticationType.tsx +++ b/app/src/pages/ODD/ConnectViaWifi/SelectAuthenticationType.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { Flex, DIRECTION_COLUMN } from '@opentrons/components' diff --git a/app/src/pages/ODD/ConnectViaWifi/SetWifiCred.tsx b/app/src/pages/ODD/ConnectViaWifi/SetWifiCred.tsx index 0c022ea0f99..4895f698b72 100644 --- a/app/src/pages/ODD/ConnectViaWifi/SetWifiCred.tsx +++ b/app/src/pages/ODD/ConnectViaWifi/SetWifiCred.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useTranslation } from 'react-i18next' import { Flex, DIRECTION_COLUMN } from '@opentrons/components' diff --git a/app/src/pages/ODD/ConnectViaWifi/WifiConnectStatus.tsx b/app/src/pages/ODD/ConnectViaWifi/WifiConnectStatus.tsx index f20e32ccb01..f00e61c6471 100644 --- a/app/src/pages/ODD/ConnectViaWifi/WifiConnectStatus.tsx +++ b/app/src/pages/ODD/ConnectViaWifi/WifiConnectStatus.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { Flex, DIRECTION_COLUMN, SPACING } from '@opentrons/components' diff --git a/app/src/pages/ODD/ConnectViaWifi/__tests__/ConnectViaWifi.test.tsx b/app/src/pages/ODD/ConnectViaWifi/__tests__/ConnectViaWifi.test.tsx index 52c0e3c95dd..004ba7b0762 100644 --- a/app/src/pages/ODD/ConnectViaWifi/__tests__/ConnectViaWifi.test.tsx +++ b/app/src/pages/ODD/ConnectViaWifi/__tests__/ConnectViaWifi.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, it, describe, expect, beforeEach, afterEach } from 'vitest' import { MemoryRouter } from 'react-router-dom' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/pages/ODD/ConnectViaWifi/index.tsx b/app/src/pages/ODD/ConnectViaWifi/index.tsx index df153941a3b..54555b85f53 100644 --- a/app/src/pages/ODD/ConnectViaWifi/index.tsx +++ b/app/src/pages/ODD/ConnectViaWifi/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import { useState } from 'react' import { useSelector } from 'react-redux' import last from 'lodash/last' @@ -27,16 +27,15 @@ export type WifiScreenOption = | 'WifiConnectStatus' export function ConnectViaWifi(): JSX.Element { - const [selectedSsid, setSelectedSsid] = React.useState('') - const [ - selectedAuthType, - setSelectedAuthType, - ] = React.useState('wpa-psk') + const [selectedSsid, setSelectedSsid] = useState('') + const [selectedAuthType, setSelectedAuthType] = useState( + 'wpa-psk' + ) - const [currentOption, setCurrentOption] = React.useState( + const [currentOption, setCurrentOption] = useState( 'WifiList' ) - const [password, setPassword] = React.useState('') + const [password, setPassword] = useState('') const localRobot = useSelector(getLocalRobot) const robotName = localRobot?.name != null ? localRobot.name : 'no name' const list = useWifiList(robotName, WIFI_LIST_POLL_MS) diff --git a/app/src/pages/ODD/DeckConfiguration/__tests__/DeckConfiguration.test.tsx b/app/src/pages/ODD/DeckConfiguration/__tests__/DeckConfiguration.test.tsx index 9715b60b09a..796e40cb6ee 100644 --- a/app/src/pages/ODD/DeckConfiguration/__tests__/DeckConfiguration.test.tsx +++ b/app/src/pages/ODD/DeckConfiguration/__tests__/DeckConfiguration.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { MemoryRouter } from 'react-router-dom' import { vi, it, describe, expect, beforeEach } from 'vitest' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/pages/ODD/EmergencyStop/__tests__/EmergencyStop.test.tsx b/app/src/pages/ODD/EmergencyStop/__tests__/EmergencyStop.test.tsx index 2bfff093233..428e276c2e8 100644 --- a/app/src/pages/ODD/EmergencyStop/__tests__/EmergencyStop.test.tsx +++ b/app/src/pages/ODD/EmergencyStop/__tests__/EmergencyStop.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, it, describe, expect, beforeEach } from 'vitest' import { useEstopQuery } from '@opentrons/react-api-client' import { fireEvent, screen } from '@testing-library/react' diff --git a/app/src/pages/ODD/EmergencyStop/index.tsx b/app/src/pages/ODD/EmergencyStop/index.tsx index 92806b6adef..7cca1beb9fd 100644 --- a/app/src/pages/ODD/EmergencyStop/index.tsx +++ b/app/src/pages/ODD/EmergencyStop/index.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { useTranslation } from 'react-i18next' import { useNavigate } from 'react-router-dom' diff --git a/app/src/pages/ODD/InitialLoadingScreen/__tests__/InitialLoadingScreen.test.tsx b/app/src/pages/ODD/InitialLoadingScreen/__tests__/InitialLoadingScreen.test.tsx index 114b09ebb3f..e9b45591b4b 100644 --- a/app/src/pages/ODD/InitialLoadingScreen/__tests__/InitialLoadingScreen.test.tsx +++ b/app/src/pages/ODD/InitialLoadingScreen/__tests__/InitialLoadingScreen.test.tsx @@ -1,4 +1,3 @@ -import * as React from 'react' import { vi, it, describe, beforeEach, afterEach } from 'vitest' import { screen } from '@testing-library/react' diff --git a/app/src/pages/ODD/InitialLoadingScreen/index.tsx b/app/src/pages/ODD/InitialLoadingScreen/index.tsx index 9bbda2e6493..f35fad13b56 100644 --- a/app/src/pages/ODD/InitialLoadingScreen/index.tsx +++ b/app/src/pages/ODD/InitialLoadingScreen/index.tsx @@ -1,4 +1,4 @@ -import * as React from 'react' +import type * as React from 'react' import { useSelector } from 'react-redux' import { ALIGN_CENTER, diff --git a/app/src/pages/ODD/InstrumentDetail/InstrumentDetailOverflowMenu.tsx b/app/src/pages/ODD/InstrumentDetail/InstrumentDetailOverflowMenu.tsx index 5692653566a..3b589295d0b 100644 --- a/app/src/pages/ODD/InstrumentDetail/InstrumentDetailOverflowMenu.tsx +++ b/app/src/pages/ODD/InstrumentDetail/InstrumentDetailOverflowMenu.tsx @@ -1,6 +1,8 @@ import * as React from 'react' import NiceModal, { useModal } from '@ebay/nice-modal-react' import { useTranslation } from 'react-i18next' +import { createPortal } from 'react-dom' + import { ALIGN_CENTER, COLORS, @@ -15,19 +17,14 @@ import { import { SINGLE_MOUNT_PIPETTES, NINETY_SIX_CHANNEL, - FLEX_ROBOT_TYPE, - getPipetteModelSpecs, } from '@opentrons/shared-data' import { ApiHostProvider } from '@opentrons/react-api-client' import { PipetteWizardFlows } from '/app/organisms/PipetteWizardFlows' import { GripperWizardFlows } from '/app/organisms/GripperWizardFlows' -import { - DropTipWizardFlows, - useDropTipWizardFlows, -} from '/app/organisms/DropTipWizardFlows' import { FLOWS } from '/app/organisms/PipetteWizardFlows/constants' import { GRIPPER_FLOW_TYPES } from '/app/organisms/GripperWizardFlows/constants' +import { getTopPortalEl } from '/app/App/portal' import type { PipetteData, @@ -38,18 +35,24 @@ import type { interface InstrumentDetailsOverflowMenuProps { instrument: PipetteData | GripperData host: HostConfig | null + toggleDTWiz: () => void } export const handleInstrumentDetailOverflowMenu = ( instrument: InstrumentDetailsOverflowMenuProps['instrument'], - host: InstrumentDetailsOverflowMenuProps['host'] + host: InstrumentDetailsOverflowMenuProps['host'], + toggleDTWiz: () => void ): void => { - NiceModal.show(InstrumentDetailsOverflowMenu, { instrument, host }) + NiceModal.show(InstrumentDetailsOverflowMenu, { + instrument, + host, + toggleDTWiz, + }) } const InstrumentDetailsOverflowMenu = NiceModal.create( (props: InstrumentDetailsOverflowMenuProps): JSX.Element => { - const { instrument, host } = props + const { instrument, host, toggleDTWiz } = props const { t } = useTranslation('robot_controls') const modal = useModal() const [wizardProps, setWizardProps] = React.useState< @@ -66,9 +69,6 @@ const InstrumentDetailsOverflowMenu = NiceModal.create( modal.remove() }, } - const { showDTWiz, toggleDTWiz } = useDropTipWizardFlows() - const pipetteModelSpecs = - getPipetteModelSpecs((instrument as PipetteData).instrumentModel) ?? null const is96Channel = instrument?.ok && @@ -97,64 +97,62 @@ const InstrumentDetailsOverflowMenu = NiceModal.create( } } + const handleDropTip = (): void => { + toggleDTWiz() + modal.remove() + } + + // TODO(jh 09-24-24): Create an ODD-specific component that wraps MenuList with a portal. return ( - - {instrument.data.calibratedOffset?.last_modified != null ? ( - - - - - {t('recalibrate')} - - - - ) : null} - {instrument.mount !== 'extension' ? ( - - - - - {t('drop_tips')} - - - - ) : null} - + {createPortal( + + {instrument.data.calibratedOffset?.last_modified != null ? ( + + + + + {t('recalibrate')} + + + + ) : null} + {instrument.mount !== 'extension' ? ( + + + + + {t('drop_tips')} + + + + ) : null} + , + getTopPortalEl() + )} {wizardProps != null && 'mount' in wizardProps ? ( ) : null} {wizardProps != null && !('mount' in wizardProps) ? ( ) : null} - {showDTWiz && - instrument.mount !== 'extension' && - pipetteModelSpecs != null ? ( - - ) : null} ) } diff --git a/app/src/pages/ODD/InstrumentDetail/__tests__/InstrumentDetail.test.tsx b/app/src/pages/ODD/InstrumentDetail/__tests__/InstrumentDetail.test.tsx index 1bfc5e53c00..f9efce16e0d 100644 --- a/app/src/pages/ODD/InstrumentDetail/__tests__/InstrumentDetail.test.tsx +++ b/app/src/pages/ODD/InstrumentDetail/__tests__/InstrumentDetail.test.tsx @@ -1,4 +1,3 @@ -import React from 'react' import { vi, it, describe, expect, beforeEach, afterEach } from 'vitest' import { screen } from '@testing-library/react' import { useParams } from 'react-router-dom' @@ -11,18 +10,31 @@ import { InstrumentDetail } from '..' import { useGripperDisplayName, usePipetteModelSpecs, -} from '/app/resources/instruments/hooks' +} from '/app/local-resources/instruments' import { useIsOEMMode } from '/app/resources/robot-settings/hooks' +import { + DropTipWizardFlows, + useDropTipWizardFlows, +} from '/app/organisms/DropTipWizardFlows' import type { Instruments } from '@opentrons/api-client' - +import type * as SharedData from '@opentrons/shared-data' + +vi.mock('@opentrons/shared-data', async importOriginal => { + const actual = await importOriginal() + return { + ...actual, + getPipetteModelSpecs: vi.fn(), + } +}) vi.mock('@opentrons/react-api-client') vi.mock('react-router-dom', () => ({ useParams: vi.fn(), useNavigate: vi.fn(), })) -vi.mock('/app/resources/instruments/hooks') +vi.mock('/app/local-resources/instruments') vi.mock('/app/resources/robot-settings/hooks') +vi.mock('/app/organisms/DropTipWizardFlows') const render = () => { return renderWithProviders(, { @@ -98,6 +110,11 @@ describe('InstrumentDetail', () => { vi.mocked(useGripperDisplayName).mockReturnValue('mockGripper') vi.mocked(useParams).mockReturnValue({ mount: 'left' }) vi.mocked(useIsOEMMode).mockReturnValue(false) + vi.mocked(useDropTipWizardFlows).mockReturnValue({ + toggleDTWiz: () => null, + showDTWiz: false, + }) + vi.mocked(DropTipWizardFlows).mockReturnValue(
MOCK_DROP_TIP_WIZ
) }) afterEach(() => { diff --git a/app/src/pages/ODD/InstrumentDetail/__tests__/InstrumentDetailOverflowMenu.test.tsx b/app/src/pages/ODD/InstrumentDetail/__tests__/InstrumentDetailOverflowMenu.test.tsx index 385e042bbb5..2137b52ed8f 100644 --- a/app/src/pages/ODD/InstrumentDetail/__tests__/InstrumentDetailOverflowMenu.test.tsx +++ b/app/src/pages/ODD/InstrumentDetail/__tests__/InstrumentDetailOverflowMenu.test.tsx @@ -1,4 +1,3 @@ -import React from 'react' import NiceModal from '@ebay/nice-modal-react' import { fireEvent, screen } from '@testing-library/react' import { vi, it, describe, expect, beforeEach, afterEach } from 'vitest' @@ -11,9 +10,7 @@ import { handleInstrumentDetailOverflowMenu } from '../InstrumentDetailOverflowM import { useNotifyCurrentMaintenanceRun } from '/app/resources/maintenance_runs' import { PipetteWizardFlows } from '/app/organisms/PipetteWizardFlows' import { GripperWizardFlows } from '/app/organisms/GripperWizardFlows' -import { useDropTipWizardFlows } from '/app/organisms/DropTipWizardFlows' -import type { Mock } from 'vitest' import type { PipetteData, GripperData, @@ -31,7 +28,6 @@ vi.mock('@opentrons/shared-data', async importOriginal => { vi.mock('/app/resources/maintenance_runs') vi.mock('/app/organisms/PipetteWizardFlows') vi.mock('/app/organisms/GripperWizardFlows') -vi.mock('/app/organisms/DropTipWizardFlows') const MOCK_PIPETTE = { mount: 'left', @@ -103,13 +99,18 @@ const MOCK_GRIPPER = { } as GripperData const MOCK_HOST: HostConfig = { hostname: 'TEST_HOST' } +const mockToggleDTWiz = vi.fn() const render = (pipetteOrGripper: PipetteData | GripperData) => { return renderWithProviders(