5 FSR-402 Force Sensing Projects for HCI Research and Human Sensing
Build pressure-aware systems: seating posture monitor, musical expression controller, accessibility switch, smart door handle, and grip force trainer
The FSR-402 (Force Sensing Resistor) is a thin, flexible polymer thick film device that decreases resistance as force is applied across its surface. Unlike load cells that measure weight precisely, FSR-402 is designed for qualitative force detection — it tells you whether someone is touching something, how hard they are pressing, and whether the pressure is increasing or decreasing. The sensor has a 0.1N to 100N force range, responds in milliseconds, and requires no special libraries — just an analog input and a 10kΩ pull-down resistor.
In HCI research, FSR-402 is used for seating posture detection, foot pressure analysis, musical expression control, and accessibility switches because it can detect physical human presence without requiring the person to do anything specific — just sit, stand, or press naturally.
In this guide, we build five human-sensing projects that translate physical pressure into interactive system responses.
Topics covered: FSR voltage-divider circuit, analog calibration, threshold-based state machines, MIDI velocity mapping, seating posture analysis, accessibility switch design, Arduino Nano analog read.
What You'll Need
- FSR-402 (×1-4 depending on project)
- Arduino Nano or Uno (×1)
- 10kΩ resistor (×1, for voltage divider)
- LED strip WS2812B (×1, for posture/visualization projects)
- MIDI-compatible circuit (for musical projects)
- Buzzer (×1, for accessibility switch)
- Servo motor SG90 (×1, for door handle project)
- Jumper wires and breadboard
- USB cable for programming
How FSR-402 Works
Technical Principle
FSR-402 consists of two layers separated by a spacer. The active zone is a force-sensitive conductive polymer whose resistance varies inversely with applied force. At rest (no force), resistance is typically above 1MΩ. As force increases, resistance drops: at 10N it is around 10kΩ, and at 100N it can fall below 1kΩ. Because the relationship is non-linear and highly variable between individual sensors, FSR is used for relative force detection rather than absolute weight measurement.
Wiring Diagram (Voltage Divider)
Arduino FSR-402
A0 ─── One FSR lead
5V ─── Other FSR lead (with 10kΩ to GND as pull-down)
Actual circuit:
5V ─── FSR ─── A0 ─── 10kΩ ─── GND
Calibration Considerations
FSR-402 sensors have significant part-to-part variation (up to ±30%). Each sensor must be calibrated individually. The ADC reading maps to force as:
Force (N) ≈ 10000 / (ADC_value) // Rough approximation only
For research applications, always calibrate against a known reference and report your calibration equation.
Important Notes
| Consideration | Detail |
|---|---|
| Hysteresis | FSR-402 shows different resistance when force is increasing vs decreasing — allow settling time |
| Temperature | Operating range 0–70°C; performance degrades outside this range |
| Creep | Under sustained force, the polymer slowly deforms — not suitable for continuous load monitoring |
| Response time | < 1ms response, suitable for real-time interaction |
| Force range | 0.1N (barely perceptible) to 100N (firm press) |
Basic Example Code
// WF1 Run #051 - Basic FSR Force Reading
#define FSR_PIN A0
#define PULL_DOWN_OHM 10000
void setup() {
Serial.begin(115200);
}
void loop() {
int adcRaw = analogRead(FSR_PIN);
// Voltage divider: Vout = 5V * R / (R_FSR + 10000)
// ADC maps 0-1023 to 0-5V, so:
// R_FSR = (5V * 10000) / (Vout) - 10000
float Vout = adcRaw * (5.0 / 1023.0);
float R_fsr = (5.0 * PULL_DOWN_OHM / Vout) - PULL_DOWN_OHM;
// Convert to approximate force (N) — calibration required for accuracy
float forceN = 10000.0 / max(R_fsr, 100.0); // Avoid divide by zero
Serial.print("ADC: ");
Serial.print(adcRaw);
Serial.print(" R: ");
Serial.print(R_fsr);
Serial.print(" ohm ~Force: ");
Serial.print(forceN);
Serial.println(" N");
delay(100);
}
Project 1: HCI Seating Posture Monitor
Goal: Detect whether a person is sitting, leaning forward, or absent from a chair, using pressure distribution from four FSR-402 sensors placed under the seat cushion
Hardware
- FSR-402 (×4)
- Arduino Nano (×1)
- 10kΩ resistor (×4)
- WS2812B LED strip (×12, arranged in a row)
- Power supply 5V/2A
Code
// WF1 Run #051 - Project 1: Seating Posture Monitor
#include <FastLED.h>
#define NUM_ZONES 4
const int FSR_PINS[NUM_ZONES] = {A0, A1, A2, A3};
const int LED_PINS[NUM_ZONES] = {2, 3, 4, 5};
#define NO_SITTING_THRESHOLD 30 // ADC — very light or no pressure
#define LEANING_THRESHOLD 200 // ADC — partial weight
#define SITTING_THRESHOLD 500 // ADC — full weight
enum PostureState { ABSENT, SITTING_UPRIGHT, LEANING_FORWARD, UNKNOWN };
PostureState currentPosture = ABSENT;
void setup() {
Serial.begin(115200);
for (int i = 0; i < NUM_ZONES; i++) {
pinMode(FSR_PINS[i], INPUT);
}
FastLED.addLeds<WS2812B, LED_DATA_PIN, GRB>(leds, NUM_LEDS);
}
int readFSR(int pin) {
int sum = 0;
for (int i = 0; i < 10; i++) sum += analogRead(pin);
return sum / 10; // Average 10 readings to reduce noise
}
PostureState determinePosture(int readings[]) {
int total = 0;
int frontWeight = 0;
int backWeight = 0;
for (int i = 0; i < NUM_ZONES; i++) {
total += readings[i];
if (i < 2) frontWeight += readings[i]; // Front sensors
else backWeight += readings[i]; // Back sensors
}
if (total < NO_SITTING_THRESHOLD * NUM_ZONES) return ABSENT;
if (frontWeight > backWeight * 1.5) return LEANING_FORWARD;
if (total > SITTING_THRESHOLD * NUM_ZONES) return SITTING_UPRIGHT;
return UNKNOWN;
}
void loop() {
int readings[NUM_ZONES];
for (int i = 0; i < NUM_ZONES; i++) {
readings[i] = readFSR(FSR_PINS[i]);
}
PostureState posture = determinePosture(readings);
// Visual feedback via LED strip
for (int i = 0; i < NUM_LEDS; i++) leds[i] = CRGB(0, 0, 0);
if (posture == ABSENT) {
// All off
} else if (posture == SITTING_UPRIGHT) {
fill_solid(leds, NUM_LEDS, CRGB(0, 50, 0)); // Green
} else if (posture == LEANING_FORWARD) {
fill_solid(leds, NUM_LEDS, CRGB(50, 30, 0)); // Yellow-amber
} else {
fill_solid(leds, NUM_LEDS, CRGB(30, 0, 0)); // Red
}
FastLED.show();
// Serial output for research logging
Serial.print("Posture: ");
Serial.print(posture == ABSENT ? "ABSENT" :
posture == SITTING_UPRIGHT ? "UPRIGHT" :
posture == LEANING_FORWARD ? "LEANING" : "UNKNOWN");
Serial.print(" | Readings: ");
for (int i = 0; i < NUM_ZONES; i++) {
Serial.print(readings[i]);
Serial.print(" ");
}
Serial.println();
delay(200);
}
Project 2: Musical Expression Controller
Goal: Convert foot pressure from an FSR-402 mat into MIDI velocity messages for real-time musical expression control
Hardware
- FSR-402 (×1, large format or mat-style)
- Arduino Nano (×1)
- 10kΩ resistor (×1)
- 5-pin MIDI DIN connector (×1)
- 220Ω resistor (×1, for MIDI TX protection)
- WS2812B LED strip (×8, for visual pressure feedback)
Code
// WF1 Run #051 - Project 2: FSR-to-MIDI Expression Controller
// Sends MIDI Note On/CC messages based on foot pressure
#define FSR_PIN A0
#define LED_DATA_PIN 6
#define NUM_LEDS 8
#define MIDI_CHANNEL 1
#define NOTE_NUMBER 48 // C3
#define MIDI_CC 74 // CC for modulation
#define PRESSURE_THRESHOLD 50 // Minimum pressure to trigger note
#define PEAK_TIMEOUT_MS 300 // Time between note triggers
CRGB leds[NUM_LEDS];
bool noteIsOn = false;
unsigned long lastNoteTime = 0;
int peakPressure = 0;
void setup() {
Serial.begin(31250); // MIDI baud rate
FastLED.addLeds<WS2812B, LED_DATA_PIN, GRB>(leds, NUM_LEDS);
Serial.write(0xFE); // Active sensing — keep connection alive
}
int readFSR() {
int sum = 0;
for (int i = 0; i < 5; i++) sum += analogRead(FSR_PIN);
return sum / 5;
}
void midiNoteOn(int cmd, int note, int velocity) {
Serial.write(cmd | MIDI_CHANNEL);
Serial.write(note);
Serial.write(constrain(velocity, 0, 127));
}
void midiCC(int cc, int value) {
Serial.write(0xB0 | MIDI_CHANNEL); // CC message on channel 1
Serial.write(cc);
Serial.write(constrain(value, 0, 127));
}
void updateLED(int pressure) {
int litLEDs = map(constrain(pressure, 0, 1023), 0, 1023, 0, NUM_LEDS);
for (int i = 0; i < NUM_LEDS; i++) {
if (i < litLEDs) {
leds[i] = CRGB(0, map(i, 0, NUM_LEDS, 50, 200), 0); // Green gradient
} else {
leds[i] = CRGB(0, 0, 0);
}
}
FastLED.show();
}
void loop() {
int pressure = readFSR();
// Update visual feedback
updateLED(pressure);
// Send continuous CC for expression
int ccValue = map(pressure, 0, 1023, 0, 127);
midiCC(MIDI_CC, ccValue);
// Trigger note on threshold crossing
if (pressure > PRESSURE_THRESHOLD && !noteIsOn) {
// Map pressure to velocity (0-1023 → 1-127)
int velocity = map(pressure, PRESSURE_THRESHOLD, 1023, 1, 127);
midiNoteOn(0x90, NOTE_NUMBER, velocity); // Note On
noteIsOn = true;
lastNoteTime = millis();
peakPressure = pressure;
} else if (pressure > peakPressure) {
// Track peak pressure during note sustain
peakPressure = pressure;
int velocity = map(peakPressure, PRESSURE_THRESHOLD, 1023, 1, 127);
midiNoteOn(0x90, NOTE_NUMBER, velocity); // Update velocity
}
// Release note when pressure drops
if (noteIsOn && pressure < PRESSURE_THRESHOLD) {
midiNoteOn(0x80, NOTE_NUMBER, 0); // Note Off
noteIsOn = false;
peakPressure = 0;
}
// Safety timeout — prevent stuck notes
if (noteIsOn && (millis() - lastNoteTime) > 2000) {
midiNoteOn(0x80, NOTE_NUMBER, 0);
noteIsOn = false;
}
delay(10);
}
Project 3: Accessibility Touch Switch
Goal: Replace mechanical switches for users with limited motor control — even the lightest touch on an FSR-402 pad triggers a reliable activation signal
Hardware
- FSR-402 (×1)
- Arduino Nano (×1)
- 10kΩ resistor (×1)
- Relay module or optocoupler (×1, for connecting to external device)
- LED indicator (×1)
Code
// WF1 Run #051 - Project 3: Accessibility Touch Switch
#define FSR_PIN A0
#define RELAY_PIN 8
#define LED_PIN 13
#define ACTIVATION_FORCE 20 // ADC threshold — very light touch
#define DEBOUNCE_MS 150
bool switchState = false;
unsigned long lastActivation = 0;
void setup() {
Serial.begin(9600);
pinMode(RELAY_PIN, OUTPUT);
pinMode(LED_PIN, OUTPUT);
digitalWrite(RELAY_PIN, LOW);
digitalWrite(LED_PIN, LOW);
}
int readFSR() {
int sum = 0;
for (int i = 0; i < 5; i++) sum += analogRead(FSR_PIN);
return sum / 5;
}
void activate() {
if (millis() - lastActivation < DEBOUNCE_MS) return;
switchState = !switchState;
digitalWrite(RELAY_PIN, switchState ? HIGH : LOW);
digitalWrite(LED_PIN, switchState ? HIGH : LOW);
Serial.println(switchState ? "SWITCH ON" : "SWITCH OFF");
lastActivation = millis();
}
void loop() {
int pressure = readFSR();
if (pressure > ACTIVATION_FORCE) {
activate();
delay(300); // Prevent rapid re-triggering while finger is on sensor
}
delay(50);
}
Project 4: Smart Door Handle Presence Detection
Goal: Detect whether a person is gripping the door handle before unlocking — useful for security systems or smart home automation
Hardware
- FSR-402 (×1, small format)
- Arduino Nano (×1)
- 10kΩ resistor (×1)
- Servo motor SG90 (×1, for physical confirmation feedback)
- LED (×1)
Code
// WF1 Run #051 - Project 4: Door Handle Presence Detection
#define FSR_PIN A0
#define SERVO_PIN 5
#define LED_PIN 13
#define GRIP_THRESHOLD 80 // ADC — light grip detection
#define UNGRIP_THRESHOLD 30 // ADC — released
#include <Servo.h>
Servo lockServo;
enum GripState { RELEASED, GRIPPED };
GripState currentGrip = RELEASED;
bool doorIsLocked = true;
void setup() {
Serial.begin(9600);
pinMode(FSR_PIN, INPUT);
pinMode(LED_PIN, OUTPUT);
lockServo.attach(SERVO_PIN);
lockServo.write(0); // Locked position
}
int readFSR() {
int sum = 0;
for (int i = 0; i < 5; i++) sum += analogRead(FSR_PIN);
return sum / 5;
}
void lockDoor() {
if (!doorIsLocked) {
lockServo.write(0); // Locked
doorIsLocked = true;
digitalWrite(LED_PIN, HIGH); // Red LED = locked
Serial.println("DOOR LOCKED");
}
}
void unlockDoor() {
if (doorIsLocked) {
lockServo.write(90); // Unlocked
doorIsLocked = false;
digitalWrite(LED_PIN, LOW); // Green LED = unlocked
Serial.println("DOOR UNLOCKED");
}
}
void loop() {
int pressure = readFSR();
if (pressure > GRIP_THRESHOLD && currentGrip == RELEASED) {
currentGrip = GRIPPED;
Serial.print("GRIP DETECTED: ");
Serial.println(pressure);
// Light grip does not unlock — requires a second grip within 3 seconds
// This is a simple anti-accidental activation measure
} else if (pressure < UNGRIP_THRESHOLD && currentGrip == GRIPPED) {
currentGrip = RELEASED;
Serial.println("GRIP RELEASED");
// Optional: auto-lock after timeout
// lockDoor();
}
delay(50);
}
Project 5: Robotic Grip Force Trainer
Goal: Train a robotic gripper to apply consistent force by monitoring FSR-402 feedback — the gripper releases when force matches the target range
Hardware
- FSR-402 (×1)
- Arduino Nano (×1)
- 10kΩ resistor (×1)
- Servo motor SG90 (×1)
- LED (×2: green for target, red for over-force)
Code
// WF1 Run #051 - Project 5: Grip Force Trainer
#include <Servo.h>
#define FSR_PIN A0
#define SERVO_PIN 5
#define GREEN_LED_PIN 3
#define RED_LED_PIN 4
#define TARGET_FORCE_MIN 200 // ADC — target force range
#define TARGET_FORCE_MAX 400
#define GRIP_ANGLE 90 // Servo angle for gripping
#define RELEASE_ANGLE 0 // Servo angle for releasing
Servo gripperServo;
bool isGripping = false;
void setup() {
Serial.begin(9600);
pinMode(FSR_PIN, INPUT);
pinMode(GREEN_LED_PIN, OUTPUT);
pinMode(RED_LED_PIN, OUTPUT);
gripperServo.attach(SERVO_PIN);
gripperServo.write(RELEASE_ANGLE);
Serial.println("Grip Force Trainer Ready");
Serial.print("Target range: ");
Serial.print(TARGET_FORCE_MIN);
Serial.print(" - ");
Serial.print(TARGET_FORCE_MAX);
Serial.println(" ADC");
}
int readFSR() {
int sum = 0;
for (int i = 0; i < 5; i++) sum += analogRead(FSR_PIN);
return sum / 5;
}
void setGrip(bool grip) {
isGripping = grip;
gripperServo.write(grip ? GRIP_ANGLE : RELEASE_ANGLE);
Serial.println(grip ? "GRIP ENGAGED" : "GRIP RELEASED");
}
void loop() {
int force = readFSR();
// Visual feedback
if (force >= TARGET_FORCE_MIN && force <= TARGET_FORCE_MAX) {
digitalWrite(GREEN_LED_PIN, HIGH); // In target range
digitalWrite(RED_LED_PIN, LOW);
} else if (force > TARGET_FORCE_MAX) {
digitalWrite(RED_LED_PIN, HIGH); // Too much force
digitalWrite(GREEN_LED_PIN, LOW);
if (isGripping) setGrip(false); // Release if over-force
} else {
digitalWrite(GREEN_LED_PIN, LOW);
digitalWrite(RED_LED_PIN, LOW);
if (!isGripping) setGrip(true); // Grip if too little force
}
Serial.print("Force: ");
Serial.print(force);
Serial.print(" | Status: ");
Serial.println(
force >= TARGET_FORCE_MIN && force <= TARGET_FORCE_MAX ? "TARGET OK" :
force > TARGET_FORCE_MAX ? "OVER FORCE — RELEASING" :
"UNDER FORCE — GRIPPING"
);
delay(100);
}
Troubleshooting
| Problem | Cause | Fix |
|---|---|---|
| FSR reads 0 or max always | Wiring error — one lead disconnected | Check both FSR leads are connected to the voltage divider |
| Readings jump randomly | FSR lead wires are fragile | Solder leads or use a connector with strain relief |
| Sensor triggers without touch | High ambient noise or floating input | Add 100kΩ pull-down resistor on signal line |
| Force calibration inconsistent | FSR-402 has ±30% part variation | Individual calibration required; log your own calibration curve |
| Servo jitter when gripping | Insufficient current from Arduino | Power servo from external 5V supply, share GND with Arduino |
| MIDI not received | MIDI baud rate wrong | Must use 31250 baud, not 9600 |
Start Here
Affiliate disclosure: As an Amazon Associate, I earn from qualifying purchases.
The right parts make the difference:
FSR-402 force sensing resistor — the standard force-sensing resistor for HCI research and human presence detection. Thin, flexible, and reliable for seating, grip, and touch applications.
Arduino Nano CH340 — compact breadboard-compatible microcontroller. Ideal for multi-sensor FSR projects where space and analog inputs are at a premium.
MIDI DIN connector 5-pin — for building MIDI expression controllers. Required for connecting Arduino serial MIDI to standard MIDI equipment.
Servo motor SG90 — the standard micro servo for gripper projects, door locks, and robotic grip force trainers.
Next Step: Human Sensing Design for Your Specific Application
If this guide gave you a clear direction for your human-sensing project — but you need a custom design tailored to your specific user population, physical environment, and interaction requirements — I can help you design that.
I offer a personalized interactive device design guide at Fiverr:
👉 https://www.fiverr.com/phd_hfchang/generate-an-arduino-interactive-prototypef
What you get:
- FSR sensor selection and placement optimized for your specific use case
- Calibration methodology for your target user population
- Interaction logic with hysteresis and debouncing tuned for real-world conditions
- Testing protocol with pass/fail criteria for reliability validation
Tags: Arduino, FSR-402, Force Sensor, HCI, Human Sensing, Posture Detection, MIDI, Accessibility, Robotics, Gripper, Interactive



Top comments (0)