Python interface for FlexiTac tactile sensors. Allows you to easily flash Arduino firmware and read framed sensor data over serial.
Defaults target the standard FlexiTac 12x32 sensor (12 rows wired to mux
channels 4-15). Override --rows, --cols, and --mux-offset for variants.
Serial port permissions apply to both flashing and reading. If you hit
Permission denied with either flexitac-flash or read tools like
flexitac-stream/flexitac-heatmap, see the note in flexitac-flash.
pyflexitac_tutorial_compressed.mp4
# recommended
uv sync # core deps only
uv sync --extra examples # include matplotlib for the heatmap CLI
uv sync --extra dev # dev tooling (ruff, mypy, pytest)
# or with pip
pip install flexitac
pip install 'flexitac[examples]' # include matplotlibfrom flexitac import FlexiTacSensor
with FlexiTacSensor("/dev/ttyUSB0") as sensor: # rows=12, cols=32 by default
for frame in sensor:
print(frame.normalized.shape, frame.normalized.max())sensor.read() returns a FlexiTacFrame(seq, timestamp_s, raw, normalized).
The first read auto-calibrates by collecting init_frames (default 30) and
storing the per-pixel median as the baseline. Call sensor.calibrate() to
recalibrate.
All CLI commands are installed as entry points:
flexitac-stream --port /dev/ttyUSB0 # stream frames & print stats
flexitac-heatmap --port /dev/ttyUSB0 # live matplotlib heatmap
flexitac-flash # flash firmware (auto-detects board)
flexitac-find-port # identify sensor port by unplug
# with uv (no install needed)
uv run flexitac-stream --port /dev/ttyUSB0
uv run flexitac-heatmap --port /dev/ttyUSB0Not sure which /dev/tty* your sensor is on? Run:
flexitac-find-portIt snapshots ports, asks you to unplug the sensor, then reports whichever port disappeared.
Stream frames and print FPS / signal stats:
flexitac-stream --port /dev/ttyUSB0 --frames 3000frame= 10 fps= 85.3 raw_max=104 norm_max=0.000
frame= 20 fps= 87.1 raw_max=109 norm_max=0.123
What to check:
fpsstabilizes near your expected rate (~100+ fps for a 12x32 sensor at 2 Mbps). If it's 0 or you getTimeoutError, the firmware isn't sending framed data -- confirm--rows/--cols/--baudmatch what you flashed.raw_maxis in[0, 255]and changes when you press the sensor. A flat 0 or flat 255 indicates a wiring issue, not a flashing issue.norm_maxrises toward 1.0 under contact and stays near 0 at rest.
Live heatmap visualization (requires matplotlib -- install with the examples extra):
flexitac-heatmap --port /dev/ttyUSB0Press the sensor pad -- bright spots should track your touch.
Requires arduino-cli.
First-time setup:
brew install arduino-cli
# or use the upstream install script:
curl -fsSL https://raw.githubusercontent.com/arduino/arduino-cli/master/install.sh | sh
# then install the Arduino AVR core
arduino-cli core update-index
arduino-cli core install arduino:avr # AVR core for Uno/Nano/Mega/etc.# auto-detects the port + FQBN if exactly one Arduino is plugged in
flexitac-flash
# override geometry / wiring for non-standard sensors
flexitac-flash --rows 16 --cols 32 --mux-offset 0
# if unable to autodetect, pass --port and --fqbn explicitly:
flexitac-flash --port /dev/ttyUSB0 --fqbn arduino:avr:unoIf flashing fails with a serial Permission denied error:
-
On Linux, add your user to the serial group, then re-login:
sudo usermod -aG dialout "$USER" -
On macOS, permissions are usually managed automatically, but if needed you can grant temporary read/write access to the device (replace with your actual port):
sudo chmod a+rw /dev/tty.usbmodem*
As a last-resort fallback, you can run sudo chmod 666 /dev/tty...,
but this is overly permissive.
Defaults: rows=12, cols=32, baud=2000000, mux-offset=4 (standard
FlexiTac 12x32 sensor wired to mux channels 4-15). The firmware is generated
from flexitac/firmware/template.ino by substituting ROW_COUNT,
COLUMN_COUNT, BAUD_RATE, and MUX_CHANNEL_OFFSET. To customize pin
assignments, edit the template directly.
Each frame: marker 0xAA 0x55 followed by rows * cols uint8 ADC samples,
streamed continuously at the configured baud rate.
uv sync --extra dev
make format # ruff format + autofix
make static-checks # ruff + mypy
make test # pytestIf you use this software in research or publications, please cite the repository. On GitHub, use Cite this repository.
@misc{maa_pyflexitac_2026,
author = {Maa, Wesley},
title = {{PyFlexiTac}: Python runtime and flashing tools for FlexiTac tactile sensors},
year = {2026},
howpublished = {\url{https://github.com/WT-MM/PyFlexiTac}},
note = {Version 0.3.1},
}- Binghao Huang, for the Arduino firmware template this project's firmware is derived from.
- Naian Tao, for leading the LeFlexiTac project that motivated this code.