Skip to content

Add offset setting, add type hints #22

New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Merged
merged 15 commits into from
Nov 24, 2021
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
41 changes: 32 additions & 9 deletions adafruit_vl6180x.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,10 +25,17 @@
https://github.com/adafruit/circuitpython/releases
* Adafruit's Bus Device library: https://github.com/adafruit/Adafruit_CircuitPython_BusDevice
"""
import struct
from micropython import const

from adafruit_bus_device import i2c_device

try:
import typing # pylint: disable=unused-import
from busio import I2C
except ImportError:
pass


__version__ = "0.0.0-auto.0"
__repo__ = "https://github.com/adafruit/Adafruit_CircuitPython_VL6180X.git"
Expand All @@ -49,6 +56,7 @@
_VL6180X_REG_RESULT_RANGE_VAL = const(0x062)
_VL6180X_REG_RESULT_RANGE_STATUS = const(0x04D)
_VL6180X_REG_RESULT_INTERRUPT_STATUS_GPIO = const(0x04F)
_VL6180X_REG_SYSRANGE_PART_TO_PART_RANGE_OFFSET = const(0x024)

# User-facing constants:
ALS_GAIN_1 = const(0x06)
Expand Down Expand Up @@ -85,15 +93,18 @@ class VL6180X:
default value will be assumed.
"""

def __init__(self, i2c, address=_VL6180X_DEFAULT_I2C_ADDR):
def __init__(
self, i2c: I2C, address: int = _VL6180X_DEFAULT_I2C_ADDR, offset: int = 0
) -> None:
self._device = i2c_device.I2CDevice(i2c, address)
if self._read_8(_VL6180X_REG_IDENTIFICATION_MODEL_ID) != 0xB4:
raise RuntimeError("Could not find VL6180X, is it connected and powered?")
self._load_settings()
self._write_8(_VL6180X_REG_SYSTEM_FRESH_OUT_OF_RESET, 0x00)
self.offset = offset

@property
def range(self):
def range(self) -> int:
"""Read the range of an object in front of sensor and return it in mm."""
# wait for device to be ready for range measurement
while not self._read_8(_VL6180X_REG_RESULT_RANGE_STATUS) & 0x01:
Expand All @@ -109,7 +120,19 @@ def range(self):
self._write_8(_VL6180X_REG_SYSTEM_INTERRUPT_CLEAR, 0x07)
return range_

def read_lux(self, gain):
@property
def offset(self) -> int:
"""Read and sets the manual offset for the sensor, in millimeters"""
return self._offset

@offset.setter
def offset(self, offset: int) -> None:
self._write_8(
_VL6180X_REG_SYSRANGE_PART_TO_PART_RANGE_OFFSET, struct.pack("b", offset)[0]
)
self._offset = offset

def read_lux(self, gain: int) -> float:
"""Read the lux (light value) from the sensor and return it. Must
specify the gain value to use for the lux reading:
- ALS_GAIN_1 = 1x
Expand Down Expand Up @@ -164,7 +187,7 @@ def read_lux(self, gain):
return lux

@property
def range_status(self):
def range_status(self) -> int:
"""Retrieve the status/error from a previous range read. This will
return a constant value such as:

Expand All @@ -182,7 +205,7 @@ def range_status(self):
"""
return self._read_8(_VL6180X_REG_RESULT_RANGE_STATUS) >> 4

def _load_settings(self):
def _load_settings(self) -> None:
# private settings from page 24 of app note
self._write_8(0x0207, 0x01)
self._write_8(0x0208, 0x01)
Expand Down Expand Up @@ -238,12 +261,12 @@ def _load_settings(self):
self._write_8(0x0014, 0x24) # Configures interrupt on 'New Sample
# Ready threshold event'

def _write_8(self, address, data):
def _write_8(self, address: int, data: int) -> None:
# Write 1 byte of data from the specified 16-bit register address.
with self._device:
self._device.write(bytes([(address >> 8) & 0xFF, address & 0xFF, data]))

def _write_16(self, address, data):
def _write_16(self, address: int, data: int) -> None:
# Write a 16-bit big endian value to the specified 16-bit register
# address.
with self._device as i2c:
Expand All @@ -258,15 +281,15 @@ def _write_16(self, address, data):
)
)

def _read_8(self, address):
def _read_8(self, address: int) -> int:
# Read and return a byte from the specified 16-bit register address.
with self._device as i2c:
result = bytearray(1)
i2c.write(bytes([(address >> 8) & 0xFF, address & 0xFF]))
i2c.readinto(result)
return result[0]

def _read_16(self, address):
def _read_16(self, address: int) -> int:
# Read and return a 16-bit unsigned big endian value read from the
# specified 16-bit register address.
with self._device as i2c:
Expand Down
35 changes: 35 additions & 0 deletions examples/vl6180x_calibrationtest.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
# SPDX-FileCopyrightText: 2018 Tony DiCola for Adafruit Industries
# SPDX-License-Identifier: MIT

# Demo of calibrating the part to part range offset per Application Note 4545
# for the VL6180X sensor

import time

import board
import busio

import adafruit_vl6180x


# Create I2C bus.
i2c = busio.I2C(board.SCL, board.SDA)

# Create sensor instance, with explicit offset of 0 to clear the system offset
sensor = adafruit_vl6180x.VL6180X(i2c, offset=0)

# Place a target at 50mm away from VL6180X Collect a number of range measurements
# with the target in place and calculate mean of the range reseults. For a
# reliable measurement, take at least 10 measurements.
measurements = []
for msmt in range(10):
range_mm = sensor.range
measurements.append(range_mm)
time.sleep(1.0)
average_msmt = sum(measurements) / 10

# Calculate the offset required:
calibration_offset = 50 - average_msmt

# Apply offset
sensor.offset = calibration_offset
3 changes: 3 additions & 0 deletions examples/vl6180x_simpletest.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,9 @@

# Create sensor instance.
sensor = adafruit_vl6180x.VL6180X(i2c)
# You can add an offset to distance measurements here (e.g. calibration)
# Swapping for the following would add a +10 millimeter offset to measurements:
# sensor = adafruit_vl6180x.VL6180X(i2c, offset=10)

# Main loop prints the range and lux every second:
while True:
Expand Down