Arduino Bike Alarm Meets Android App Via Bluetooth
by RestlessPedals in Circuits > Arduino
61 Views, 2 Favorites, 0 Comments
Arduino Bike Alarm Meets Android App Via Bluetooth

I created this Arduino-based bike alarm for my cycling trips to keep my bike secure overnight while camping. It provides peace of mind by alerting me on my phone if any movement is detected.
An older version of the Arduino alarm only had a buzzer and that made it difficult to hear it. Since I added the Android app to the project, it is much better. I can start/stop the detection from my phone, control the settings, get alerts if the battery on Arduino runs low or if communication is lost.
The instructions in this article are for the Arduino bike alarm. The Android app is available for download here.
Supplies
Arduino board
Preferably you should use Pro mini 3.3V, but you could really use any compatible board with few mentions.
3.3V vs 5V: 3.3V consumes less power which improves battery life. If you use 5V, you should use a voltage divider on the BLE module's RX, since it is not 5V tolerant.
Interrupt pins: If you use a different board, make sure to use interrupt capable pins for the ball switch and BLE module's TX. To conserve power, the board will enter sleep mode, and these two inputs must trigger the watchdog to ensure it detects all alarms and Bluetooth communications.
Ball switch: I used and recommend the SW-420, as it is highly sensitive and the code is designed to handle multiple rapid triggers. If you choose a different sensor, you may need to adjust the code accordingly. An active sensor, such as an accelerometer, is not an option for this low-power setup.
Bluetooth LE module: Tested with an HM-10 module, but could potentially work with any LE module that has the service UUID 0xFFF0, characteristic 0xFFF1 and supports notifications. Let me know in the comments if you've used different hardware.
Components
- Arduino Pro Mini 3.3V
- HM-10 BLE 4.0 Bluetooth module or compatible (Android app is hard linked to characteristics of this model - FFE0/FFE1)
- Buzzer
- SW-420 ball switch
- Mini DC-DC boost converter to 5V or similar
- resistors (1x~220Ω for buzzer, 1x~10kΩ and 2xlarge (100kΩ - 1MΩ) for reading the battery level)
- 3.7V battery
Wiring

Follow the diagram to connect the wires. You may use different pins, if you update the code accordingly, but make sure the ball switch and BLE module's TX are connected to interrupt capable pins. This is important, so the Arduino board will wake from sleep when an alarm occurred or a command from the app is sent.
Resistors R3 and R4 should be large to avoid unnecessary battery drain. I have used 1MΩ resistors. Resistor R2 should be much lower than R3/R4, around 10 KΩ is fine. If you don't have resistors, you can build the project without, but you won't receive low battery alerts on your phone. R1 is also optional, but recommended.
Communication Codes Explained
One byte commands are used to communicate with the Android app. Some are ingress (e.g. mute command coming from Android), others are egress (e.g. alarm sent to Android). Ingress commands also have acknowledgment responses, to inform Android that the command was received and executed.
Ingress:
- CMD_START and CMD_STOP to start/stop detection
- CMD_PING to test connection. Android expects to receive CMD_PONG
- CMD_BEEP simulate an alarm on Arduino without sending CMD_ALARM to Android
- CMD_READ_CONFIG send to Android the current configuration (sensibility, duration, etc)
- CMD_SAVE_CONFIG persist config variables (sensibility, duration, etc) on Arduino
- CMD_MUTE_ON, CMD_MUTE_OFF mute buzzer (not the Android alarm)
- CMD_BEACON_ON, CMD_BEACON_OFF toggle the use of beacon signals
- CMD_DURATION to configure the alarm (buzzer) duration
- CMD_LOW, CMD_MED, CMD_HIGH to configure the alarm trigger sensitivity
Egress:
- CMD_PONG response to CMD_PING
- CMD_BEACON send beacon signal to let the Android app know the system is functional
- CMD_ALARM when movement is detected
- CMD_LOW_BATTERY when device battery is low
- response to any ingress command, calculated as ingress_command - 8
Programming Arduino
We will use LowPower.h library to put the device to sleep and extend battery life. The library EEPROM.h is used to persist the selected configuration. Finally, we will connect to the Bluetooth module using SoftwareSerial.h.
First, we configure the pins. Here you can edit if you are using different wiring connections.
Here you can change the ratio between sensitivity levels. They are relative to each other, so the numbers don't really matter, but it is best to leave HIGH_SENS_FACTOR as 1 and customize the other accordingly.
You may want to tweak the low battery threshold here, according to your battery voltage.
Define config variables.
To store the configuration in memory, we need to define memory addresses for each variable. Beacon and muted are boolean variables, so they fit in one byte, thus the memory address of ADDR_MUTED is just 1 byte more than ADDR_BEACON. Sensitivity is a unsigned char, thus it can also be stored on one byte. Duration is short int, so it is stored on 2 bytes, memory address 0x3 - 0x4.
Flags are used to disable the watchdog's response during normal operations, such as processing a command or alarm, to prevent it from continuously triggering.
Define callback functions to handle interrupts. The ballMovement counter is what determines the sensitivity. Alarms will not trigger unless this variable is within an expected range.
Setting up the pins, interrupts and loading the config from EEPROM.
Parse incoming commands, execute and send ACK signals back to Android. The response codes and other egress commands are placed in a buffer and sent in one operation to prevent overwriting before Android has had a chance to read from Bluetooth.
This function sounds the buzzer in case of an alarm, or a test from Android, only if the buzzer isn't muted.
Executing the code for the first time will surely read gibberish from the EEPROM config storage. If this is the case, validateEEPROMvars() will initialize config values with defaults.
A special encoding was employed to send the config variables to Android in only 2 bytes.
The first encoded byte starts with 0x5- and carries muted and duration. Since duration takes values in the range 0-7, it fits on 3 bits (000 - 111). The fourth bit is the boolean (0 or 1) muted variable. Since a byte is made of 8 bits, the first 4 are the signature (5 in this case) and the other 4 carry the payload. So, an example for sensitivity = 2 (binary 10) and muted on (1), would make the encoded byte 0x56 (0101 for 5, 0110 for payload).
The second encoded byte starts with 0x6 and carries detection status, beacon status and sensitivity.
The Android app knows these bytes are special because no other commands start with 5 or 6.
To explain how the bit encoding works, let's look at the first byte. The operator << is the bitwise shift left operator. The right operand represents how many bits to shift to the left. The number 5 (101 in binary) shifted left by 4, becomes 1010000 in binary. This clears out the last 4 bits for the payload.
encodedByte1 = (5 << 4) | (isMuted << 3) | durationEncoded
Duration is at most 7 (111 in binary), thus it is stored on 3 bits. This is why isMuted is shifted left by 3.
The | operator is the bitwise OR and preserves bits set to 1 in all the operands. Since 0 | 1 = 1 | 0 = 1 | 1 = 1,
1010000 | 1101 = 1011101.
When the time is right, the response(s) are sent to Android using SoftwareSerial write().
The ingress commands are parsed and executed here and the ACK response is returned to signal that the command was executed.
Next, we will look at the main loop. Treat the following code as part of loop() function.
Read battery level and report to Android when the level is below the low level threshold.
Read incoming commands, if any. BLE.available() will signal if new messages were received.
Handle alarm detection.
If beacons are on, send beacon to Android. Beacons are signals that are sent regularly, even in the absence of other events, so the listening party is sure that the system is still operational.
Finally, send to Android whatever is in the buffer and sleep to save power.
Downloads
The App or the Challenge



Once your Arduino project is done, get the Android app to enjoy the benefits of your hard work. The app runs a foreground service that will monitor the connection to the Arduino device and notify on alarms, low battery or connection loss. It allows the user to remotely control the Arduino, start/stop the alarm detection, configure sensitivity or preferred ringtone, mute the alarm sound on either Arduino or Android without an impact on the actual notification.
If you don't want to use the app, I challenge you to make another Arduino project using similar hardware and change the code to allow communication between the two devices. You'll need to add a button for start/stop and optionally for beacon on/off and for mute on/off. The challenge is writing the code. The current ingress commands will become egress on the second device.
If you've made this project, write a comment to let me know useful did you find it.