- Introduction
- 1.1 Overview
- 1.2 Project Components
- 1.3 Features
- Installation
- 2.1 Required Libraries
- 2.2 Hardware Setup
- 2.3 Software Setup
- Arduino Code
- 3.1 Sensor Configuration
- 3.2 WiFi and UDP Configuration
- 3.3 Kalman Filter Setup
- 3.4 Magnetometer Calibration
- Sensor Data Update
- 4.1 Updating Sensor Data
- 4.2 Updating MPU6050 Sensor Data
- 4.3 Updating HMC5883L Sensor Data
- 4.4 Setting Motion Detection Threshold and Duration for MPU6050
- 4.5 Enabling Motion Interrupt for MPU6050
- 4.6 Initializing Kalman Filter Angles (kalAngleX, kalAngleY, kalAngleZ)
- UDP Data Sending
- 5.1 JSON Data Serialization (sensorGetJson())
- 5.2 Sending UDP Data (udpSend())
- User Interface in Blender
- 6.1 Blender Plugin Overview
- 6.2 Server Configuration Panel (SERMOTIONS_PT_server)
- 6.3 Blender Operators for Start and Save (OT_start, OT_save)
- 6.4 Blender Panel Registration (register(), unregister())
- Unreal Engine Integration
- 7.1 Plugin Integration in Unreal Engine
- 7.2 Receiving Motion Tracking Data in Unreal Engine
- Folder Structure
- 8.1 Generic Project Folder Structure
- Compiling and Uploading Arduino Code Using VS Code and PlatformIO
- 9.1 Setting Up PlatformIO in VS Code
- 9.2 Compiling and Uploading Code to Microcontroller
- Conclusion
- 10.1 Summary
- 10.2 Acknowledgments
- 10.3 License
- 10.4 Contact Information
The Motion Tracking Suit project is a comprehensive system designed to track real-time motion using Arduino-based sensors and visualize the data in Blender and Unreal Engine. The project offers a versatile solution for motion tracking that finds applications in fields like animation, game development, and virtual reality experiences.
Breadboard Schema | Breadboard Montage |
---|---|
The project comprises three main components:
-
Arduino Code: Responsible for reading sensor data from MPU6050 and HMC5883L sensors and transmitting it to the server via WiFi and UDP.
-
Python Scripts: Act as the server to receive sensor data from Arduino and send it to Blender and Unreal Engine using a custom protocol.
-
Blender and Unreal Engine Integration: Provides user interfaces in Blender and Unreal Engine for configuring server settings and visualizing the motion data.
The Motion Tracking Suit project boasts several key features:
- Real-time motion tracking using Arduino-based sensors.
- Configurable sensor settings and calibration for precise tracking.
- Seamless integration with Blender and Unreal Engine for data visualization.
- User-friendly interfaces in Blender and Unreal Engine for server configuration.
Before proceeding, make sure to install the following libraries in your Arduino IDE:
- Wire.h
- Arduino.h
- ESP8266WiFi.h
- WiFiUdp.h
- Adafruit_MPU6050.h
- Adafruit_HMC5883_U.h
- ArduinoJson.h
- Kalman.h
To set up the hardware:
- Connect the MPU6050 and HMC5883L sensors to the Arduino board following the wiring instructions provided in the Arduino code.
Upload the Arduino code to your Arduino board using the Arduino IDE.
The Arduino code configures the MPU6050 and HMC5883L sensors with specific settings.
// Sensor Configuring
Adafruit_MPU6050 mpu6050;
Adafruit_HMC5883_Unified mag = Adafruit_HMC5883_Unified(12345);
// ... (continued in the next snippet)
The code sets up the WiFi connection and UDP port for
data transmission.
// Wifi Configuration
#define WIFI_SSID "YOUR_WIFI_SSID"
#define WIFI_PASSWORD "YOUR_WIFI_PASSWORD"
// Udp Configuration
#define UDP_IP "YOUR_SERVER_IP_ADDRESS"
#define UDP_PORT 4455
#define UDP_JSON true
// ... (continued in the next snippet)
The code initializes Kalman filter instances for angle calculations.
// Kalman Filter Setup
Kalman kalmanX, kalmanY, kalmanZ; // Create the Kalman instances
// ... (continued in the next snippet)
The code calibrates the magnetometer to improve accuracy.
// Magnetometer Calibration
#define MAG0MAX 603
#define MAG0MIN -578
#define MAG1MAX 542
#define MAG1MIN -701
#define MAG2MAX 547
#define MAG2MIN -556
float magOffset[3] = {(MAG0MAX + MAG0MIN) / 2, (MAG1MAX + MAG1MIN) / 2, (MAG2MAX + MAG2MIN) / 2};
double magGain[3];
// ... (continued in the next snippet)
The code updates the sensor data, including acceleration, gyroscope, temperature, and magnetometer readings.
// Update all the IMU values
updateMPU6050();
updateHMC5883L();
// ... (continued in the next snippet)
The code reads accelerometer and gyroscope data from the MPU6050 sensor.
void updateMPU6050()
{
sensors_event_t a, g, temp;
mpu6050.getEvent(&a, &g, &temp);
accX = a.acceleration.x;
accY = a.acceleration.y;
accZ = a.acceleration.z;
tempRaw = temp.temperature;
gyroX = g.gyro.x;
gyroY = g.gyro.y;
gyroZ = g.gyro.z;
}
The code reads magnetometer data from the HMC5883L sensor.
void updateHMC5883L()
{
sensors_event_t event;
mag.getEvent(&event);
magX = event.magnetic.x;
magY = event.magnetic.y;
magZ = event.magnetic.z;
// ... (continued in the next snippet)
The code sets motion detection threshold and duration for the MPU6050 sensor.
mpu6050.setMotionDetectionThreshold(1);
mpu6050.setMotionDetectionDuration(20);
The code enables motion interrupt for the MPU6050 sensor.
mpu6050.setInterruptPinLatch(true);
mpu6050.setInterruptPinPolarity(true);
mpu6050.setMotionInterrupt(true);
The code initializes Kalman filter angles for roll, pitch, and yaw.
kalmanX.setAngle(roll); // First set roll starting angle
gyroXangle = roll;
compAngleX = roll;
kalmanY.setAngle(pitch); // Then pitch
gyroYangle = pitch;
compAngleY = pitch;
kalmanZ.setAngle(yaw); // And finally yaw
gyroZangle = yaw;
compAngleZ = yaw;
The code serializes sensor data into JSON format.
String sensorGetJson()
{
updateSensor();
StaticJsonDocument<1024> static_json_document;
// Get acceleration values
int axis[3] = {roll, pitch, yaw};
calculateAxis(axis);
static_json_document["roll"] = axis[0];
static_json_document["pitch"] = axis[1];
static_json_document["yaw"] = axis[2];
char doc_buffer[1024];
serializeJson(static_json_document, doc_buffer);
return String(doc_buffer);
}
The code sends the serialized JSON data via UDP.
void udpSend(String message, int delay_ms)
{
Udp.beginPacket(UDP_IP, UDP_PORT);
Udp.write(message.c_str());
Udp.endPacket();
Serial.printf("Sending packet: %s\n", message.c_str());
delay(delay_ms);
}
6.1 Blender Plugin Overview The Blender plugin provides a user interface for configuring server settings and controlling data streaming.
The server configuration panel allows users to set the IP address and port for data transmission.
class SERMOTIONS_PT_server(bpy.types.Panel):
bl_idname = "SERMOTIONS_PT_server"
bl_label = "Server Configuration"
bl_category = "Sermotions"
bl_space_type = "VIEW_3D"
bl_region_type = "UI"
def draw(self, context):
layout = self.layout
scene = context.scene
props = scene.props
layout.prop(props, "ip")
layout.prop(props, "port")
# ... (continued in the next snippet)
The Blender operators handle the events for starting and saving server configurations.
class SERMOTIONS_PT_server(bpy.types.Panel):
# ... (previous code)
class OT_save(bpy.types.Operator):
bl_label = "Save"
bl_idname = "sermotions.save"
def execute(self, context):
config.write()
SERMOTIONS_PT_server.status_print(self, "Server configuration is saved.")
return {"FINISHED"}
class OT_start(bpy.types.Operator):
bl_label = "Start"
bl_idname = "sermotions.start"
def execute(self, context):
global proc_server
if not proc_server.is_alive():
proc_server.start()
SERMOTIONS_PT_server.status_print(self, "Server started.")
print_debug("UDP server started.")
else:
proc_server.terminate()
SERMOTIONS_PT_server.status_print(self, "Server stopped.")
proc_server = create_process_server()
print_debug("UDP server stopped.")
return {"FINISHED"}
# ... (continued in the next snippet)
The register and unregister functions are used to add the server configuration panel to Blender's user interface.
def register():
bpy.utils.register_class(SERMOTIONS_PT_main)
bpy.utils.register_class(SERMOTIONS_PT_server)
bpy.utils.register_class(SERMOTIONS_PT_server.OT_save)
bpy.utils.register_class(SERMOTIONS_PT_server.OT_start)
bpy.types.Scene.server_status = bpy.props.StringProperty()
bpy.types.Scene.props = bpy.props.PointerProperty(type=SERMOTIONS_PT_server.PT)
def unregister():
bpy.utils.unregister_class(SERMOTIONS_PT_main)
bpy.utils.unregister_class(SERMOTIONS_PT_server)
bpy.utils.unregister_class(SERMOTIONS_PT_server.OT_save)
bpy.utils.unregister_class(SERMOTIONS_PT_server.OT_start)
del bpy.types.Scene.server_status
del bpy.types.Scene.props
The project's Unreal Engine plugin facilitates the integration of the motion tracking system.
The plugin enables Unreal Engine to receive motion tracking data and use it for various applications.
Organizing your project files in a structured manner ensures clarity and ease of management. Below is a suggested generic folder structure for your electronics project:
MotionTrackingSuit/
│
├── ArduinoCode/
│ ├── src/
│ └── lib/
│
├── BlenderPlugin/
│ ├── scripts/
│ └── icons/
│
├── UnrealPlugin/
│ ├── Source/
│ └── Resources/
│
└── Documentation/
├── Images/
│ ├── ArduinoConnections.png
│ ├── BlenderPluginInterface.png
│ ├── UnrealIntegration.png
│ └── ...
├── README.md
└── ...
To streamline the development process, you can use Visual Studio Code (VS Code) with the PlatformIO extension to compile and upload the Arduino code to your microcontroller. Here's how to set it up:
- Install Visual Studio Code.
- Install the PlatformIO IDE extension for VS Code.
- Open your project folder in VS Code.
With the PlatformIO extension installed, follow these steps to compile and upload the Arduino code to your microcontroller:
- Open the Arduino code file in VS Code.
- Click on the "Checkmark" icon in the PlatformIO toolbar to compile the code.
- Once compiled successfully,
click on the "Arrow" icon to upload the code to your microcontroller.
The Motion Tracking Suit project offers a comprehensive solution for real-time motion tracking using Arduino-based sensors. With integration in Blender and Unreal Engine, the project opens up possibilities in animation, game development, and virtual reality experiences.
We would like to express our gratitude to all contributors and collaborators who made this project possible.
This project is licensed under the MIT License.
For any inquiries or assistance, feel free to contact us at seyupaltin@gmail.com.