Computer Vision-Based Harmonic Oscillation Analysis

Computer Vision-Based Harmonic Oscillation Analysis

⭐Creative Coding Competition⭐


Introduction

A few days ago, I stumbled upon the Creative Coding Competition organized by Radu Mariescu-Istodor, someone I've been following for quite a while. I even joined his Discord server in May 2023. Inspired by his projects and tutorials, I've been recreating some of his work for a while now. I first got hooked on his machine learning tutorials using JavaScript, which I discovered on freecodecamp.org youtube channel. So, when I saw this competition, I couldn't resist giving it a shot. It was a bit out of my comfort zone, but the experience turned out to be unexpectedly rewarding. Coding became a canvas for creative expression, and I couldn't be more excited to share this newfound adventure with you today.

postscript: I actually scored the highest in the competition (46/50), here's the video reviewing my project by Radu Mariescu-Istodor Harmonic Oscillator Analyzer

The Challenge

The challenge was straightforward and enticing: Take a piece of video footage and transform it with a visual effect, using nothing but code. The only rule? There were no rules, except that your creation had to be born from lines of code.


This is my submission of the project

github.com/rudrodip/Harmonic-Oscillator-CV

Overview of the project

This project is a powerful tool designed to analyze harmonic oscillation of objects using computer vision techniques. Leveraging a variety of libraries, including OpenCV for video feed processing, SciPy for data analysis, NumPy for numerical operations, and PyQt5 for a user-friendly graphical interface, it offers a wide range of features to facilitate detailed analysis.

Key Features

  1. Bob Color Detection: The system can accurately detect the bob of any color, providing flexibility in object selection.

  2. Contour Detection Options: Choose from three contour detection methods - color-based, edge-based, or circle detection - to suit your specific tracking needs.

  3. Video Source Flexibility: Analyze harmonic oscillations from various sources, including:

    • Local Video: Process videos stored on your device for in-depth analysis.
    • Webcam: Utilize your webcam for real-time tracking and motion analysis.
  4. Display Options:

    • Main Video: Observe the original video feed.
    • Contours: Visualize the detected contours for better tracking insights.
    • Mask: Monitor the real-time masking process and make adjustments as needed.
  5. Real-time Parameter Adjustment: Easily fine-tune masking parameters in real time and save your configurations for future reference.

  6. Parameter Saving: After prediction, you can conveniently save the calculated parameters for further analysis and documentation.

  7. Frame Offset Control: Within the code, set a frame offset and utilize a built-in function to save critical data points, streamlining the analysis process.

  8. Rotation Compensation: The system intelligently compensates for video rotation, ensuring accurate analysis even when dealing with rotated footage.

  9. Prediction Capabilities:

    • Pivot Point Prediction: Accurately predict the pivot point of the bob's motion.
    • Mean Point Determination: Find the mean point of the bob's trajectory.
    • Path Analysis: Analyze the bob's path and generate predictions of its corresponding harmonic motion equation.
  10. Physical Properties Estimation: Utilizing parameters derived from the predicted harmonic motion equation, the system can estimate crucial physical properties of the system, such as the length of the string or other relevant characteristics.

This project provides a robust and versatile platform for in-depth analysis of harmonic oscillations using computer vision techniques. Its user-friendly interface, real-time adjustments, and predictive capabilities make it an invaluable tool for researchers, educators, and enthusiasts exploring dynamic systems and motion analysis.


How to use it?

It is highly advisable for you, to thoroughly go through UI section. Doing so will greatly facilitate the project setup process and enable a better grasp of the user interface (UI). Your understanding of this section is pivotal in ensuring the efficient configuration of the project and in comprehending the UI.

Prerequisites

Before you begin, ensure you have met the following requirements:

  • Python 3.6 or higher installed on your system. You can download Python from python.org.
  • Git installed (optional, but recommended for cloning the repository).

Setting Up a Virtual Environment

It's a good practice to work within a virtual environment to isolate your project's dependencies. Here's how to set up and activate a virtual environment:

  1. Open a terminal or command prompt.

  2. Clone this repository to your local machine:

git clone https://github.com/rudrodip/Harmonic-Oscillator-CV
  1. Navigate to your project directory:
cd Harmonic-Oscillator-CV
  1. Create a virtual environment (replace venv with your preferred name):
python -m venv venv
  1. Activate the virtual environment:
  • On Windows:
venv\Scripts\activate
  • On macOS and Linux:
source venv/bin/activate
  1. Your terminal prompt should now show the name of the virtual environment, indicating that it's active.

Installing Dependencies

Once you have your virtual environment set up and activated, you can install the project's dependencies:

  1. Make sure you are in your project directory with the activated virtual environment.

  2. Install the dependencies from the requirements.txt file:

pip install -r requirements.txt

Usage

  1. Run the main application script:
python app/app.py

If you find any difficulties installing or running it, feel free to contact me. I have developed this project on linux (ubuntu 22.04), if you have any issues installing packages on your machine, feel free to change the code.

Note: The OpenCV library you will be using for this project is opencv-python-headless (not opencv-python), so make sure you have the correct library installed in virtual environment, otherwise uninstall opencv-python and reinstall opencv-python-headless using this command

pip install opencv-python-headless --no-cache-dir

User Interface

Let's take a closer look at the user interface (UI) of this project, which plays a central role in interacting with the video processing and analysis. The UI design is straightforward and user-friendly, consisting of several key elements:

Buttons and Dropdowns

blog image
  1. Run: Initiates video processing after selecting a video source (file, webcam, or URL).
  2. Stop: Halts the video processing and output.
  3. Select Video: Allows the user to choose a video file for analysis.
  4. Webcam: Selects the webcam as the video source.
  5. URL: Enables video processing from a provided URL.
  6. Hide Params: Toggles the display of parameters such as pivot point, circular path, and the string connecting the pivot and the bob.

Dropdowns:

  • Display Options: Allows you to select how the video is displayed. Options include "Image Contours" (showing the detected bob with a bounding box), "Main Video" (the raw video), and "Mask" (the mask used for detecting the bob).

  • Color Detection: Provides options for how the image processor detects the bob. Options include "Color detection using HSV range" (recommended), "Edge detection," and "Circle detection."

HSV Range Sliders

blog image

These sliders are used for color detection. By adjusting the sliders, you can define the HSV (Hue, Saturation, Value) color range used by the image processor to detect the object. You can save these settings by clicking the Save HSV button.

A recommended workflow is to set the display option to "Mask," select "Color Detection," run the video, and fine-tune the sliders until you achieve optimal object detection.

Analyze Widget

blog image

The analyze widget is a crucial part of the UI. After selecting a video source and running the analysis, you can click the Estimate button. This action triggers the fitting of a curve to the data points collected by the image processor. The widget provides parameters and settings for this analysis. You can also save these parameters using the Save button.

Graph

blog image

The graph section is perhaps the most visually appealing part of the UI. It utilizes the pyqtgraph library and offers advanced graph manipulation capabilities. You can scale, zoom, pan, and save the graph in various formats (SVG, PNG, Matplotlib window, CSV). Its flexibility and interactive features make it a powerful tool for visualizing and analyzing the harmonic oscillation data.


Project Sections

The project can be divided into three major sections:

  1. Programming: In this section, I've covered the development of the user interface and the image processing pipeline, using libraries such as OpenCV and cvzone. The UI creation was straightforward, thanks to Python and PyQt5.

  2. Mathematics: In the mathematics section of the project, I utilize powerful mathematical tools to extract insights from the data

    • Curve Fitting with curve_fit: I use SciPy's curve_fit function to fit collected data points to a damped oscillation function. This helps me accurately detect and analyze harmonic motion patterns.

    • least_square for Circular Path Detection: The least_square algorithm is not only used for curve fitting but also for assessing the accuracy of circular paths. It ensures that predicted paths closely match the actual object motion, validating the precision of path predictions and harmonic motion analysis.

  3. Physics: While closely related to mathematics, the physics aspect primarily focuses on extracting physical parameters such as pendulum length and pivot point location. These parameters provide valuable insights into the object's harmonic motion.

This breakdown of the user interface and project sections should provide a clearer picture of the project's structure and functionality.


Programming

Let's dive deep 🙂

User Interface (UI)

Creating the user interface for this project was a relatively smooth process. Thanks to Python and PyQt5, I was able to design the interface without much hassle. While the UI design took some effort due to its complexity, it wasn't the most challenging part of the project.

Image Processing

  • magic 🪄

OpenCV and cvzone

Image processing is where the real magic happens in this project. OpenCV is a fantastic library, and it played a central role in handling the video feed. It allowed me to perform tasks like object detection, tracking, and image manipulation with ease.

Additionally, I leveraged cvzone for contour detection, which was particularly helpful for identifying objects based on their HSV color range. This library simplified a crucial part of the image processing pipeline.

Main Loop Workflow

The main loop of the project was the heart of the image processing section. Here's how it worked:

  1. Frame Acquisition: At the beginning of each iteration, I obtained a new frame from the video feed.

  2. Contour Detection: With the help of cvzone, I detected contours in the frame. This step was crucial for pinpointing the object of interest, often based on its distinct color range in the HSV color space.

  3. Pivot Point Estimation and Path Prediction: This step was the real technical challenge. I used the least_square algorithm from SciPy to estimate the pivot point of the oscillating object. Moreover, I employed this algorithm to predict the future path of the object. This was essential for analyzing the object's harmonic oscillation characteristics, which I'll elaborate on in the mathematics section.

  4. Data Transfer to UI: To keep the user informed, I sent relevant data, including the transformed position (transformed_x) and frame count (frame_count), back to the main user interface.

  5. Frame Rendering: The current frame was also sent to the UI for real-time rendering, allowing users to visualize the object's motion and the ongoing analysis.

  6. Repeat: These steps were executed in a continuous loop for every frame in the video feed, enabling the real-time analysis of the object's oscillations.


Mathematics

In this section, I'll take you on a mathematical journey through the key tools and techniques that drive this project, making it easier to understand.

Detecting Circular Motion with least_square

In this section, I'll walk you through the critical steps involved in using least_square to detect circular motion, even if the video is rotated. Let's dive into how I achieved this:

1. Contour Extraction

Why? To accurately detect the position of the oscillating object (the bob) in each frame.

How? I started by extracting the object's contours from each frame of the video feed. Contours are like outlines that provide precise coordinates of the object's location.

2. Calculating Residuals

Why? To assess how closely the detected path resembles a circle and understand any deviations.

How? I implemented the circle_residuals function, powered by least_square. This function calculates the differences between the observed data points and the expected points on a circle.

def circle_residuals(params, x, y):
  """
  Calculate the residuals for circle fitting.
 
  Args:
      params: Parameters of the circle (a, b, r).
      x: x-coordinates of data points.
      y: y-coordinates of data points.
 
  Returns:
      Residuals indicating the difference between the data points and the circle model.
  """
  a, b, r = params
  return np.sqrt((x - a) ** 2 + (y - b) ** 2) - r
Circle Residuals: ypredicted=x2+y2r\text{Circle Residuals: } y_\text{predicted} = \sqrt{x^2 + y^2} - r

Where:

  • predictedy\text{predicted}_y represents the predicted residuals.
  • rr is the radius of the circle.
  • xx and yy are the arrays containing the x and y coordinates of data points.

This equation calculates the residuals for fitting a circle to a set of data points.

3. Optimization

Why? To fine-tune and optimize the parameters of the rotated circle for accuracy.

How? I let least_square do its magic. It optimized the circle's parameters, including center, radius, and rotation angle. This step ensures that the predicted circular path aligns closely with the actual object path, even if there is rotation in the video.

The Result? A robust method for accurately detecting circular motion, even if the video is rotated. It's like having a compass that guides us to the truth in a sea of data.

4. Pivot Point and Rotation

Why? To account for video rotation and accurately transform the object's coordinates.

How? When I identified the circular motion, I used a straightforward method. First, I calculated the average position of the pendulum. Then, I found the tangent between the pivot point and this mean position. This allowed me to determine the precise angle of rotation, which, in turn, enabled me to accurately adjust the bob's x and y coordinates.

The Result? This entire workflow ensured that I could analyze harmonic motion, correct for video rotation, and extract precise data for further analysis.

Fitting the Harmonic Puzzle with curve_fit

At the heart of this mathematical marvel lies the curve_fit function from SciPy. This is where the magic unfolds, as I delve deep into the data to uncover the secrets of harmonic oscillations.

Why ?

Purpose: curve_fit serves as my mathematical detective, deciphering the hidden patterns within my data. Its mission? To find the perfect mathematical equation that describes the object's motion.

How it Works:

  1. Collect Data: I start by collecting data points that represent the object's position over time. These data points are like breadcrumbs left by the oscillating object.

  2. Select a Model: I choose a model that I believe fits the data. In my case, it's a under damped oscillation function. This function includes parameters such as amplitude, omega, damping coefficient, and phase, which describe the expected motion of the object.

Equation for the under-damped oscillation function

x(t)=Aeγtcos(ωt+ϕ)+Cx(t) = A e^{-\gamma t} \cos(\omega t + \phi) + C
def underdamped_harmonic_oscillator(t, A, gamma, w, phi, C):
    """
    Calculate the position of an underdamped harmonic oscillator at time t.
 
    Args:
        t: Time values.
        A: Amplitude of oscillation.
        gamma: Damping coefficient.
        f: Frequency of oscillation.
        phi: Phase angle.
        C: Constant offset.
 
    Returns:
        Position values at the given time points.
    """
    return A * np.exp(-gamma * t) * np.cos(w * t + phi) + C
  1. Let curve_fit Work Its Magic: This is where the excitement builds. I feed the data points and my chosen function (the damped oscillation model) into curve_fit. This function employs a clever optimization algorithm to adjust the model's parameters until it best matches the data.

  2. Parameter Extraction: After the optimization done, I extract the best-fit parameters. These parameters are like the missing pieces of my puzzle, allowing me to quantify the motion's amplitude, omega, damping coefficient, and phase.

x_data = data[:, 0]
y_data = data[:, 1]
 
params, _ = curve_fit(
    underdamped_harmonic_oscillator,
    x_data,
    y_data,
    p0=initial_guess,
    bounds=(lower_bounds, upper_bounds),
    maxfev=2500,
    ftol=1e-6,
)
 
# Extract the fitted parameters
A, gamma, w, phi, C = params

The Result? A beautifully fitted curve that elegantly captures the essence of harmonic motion. It's akin to assembling a puzzle and revealing the complete picture.


Physics

Now, let's explore the fascinating physics behind harmonic motion in a way that's easy to understand. Imagine we have a system—a mass attached to a spring with a damper—and it's doing some interesting things.

The Second-Order ODE: What's That?

In the realm of physics, we often describe the motion of objects using mathematical equations. In our system, we employ a second-order ordinary differential equation (ODE):

Here's the equation in its mathematical form:

md2x(t)dt2+γdx(t)dt+kx(t)=0m \frac{d^2x(t)}{dt^2} + \gamma \frac{dx(t)}{dt} + kx(t) = 0

Here's what each component means:

  • x(t)x(t) is the object's position at time tt.
  • dx(t)dt\frac{dx(t)}{dt} is its velocity, and d2x(t)dt2\frac{d^2x(t)}{dt^2} is its acceleration.
  • mm is the object's mass.
  • γ\gamma represents damping coefficient, which resists motion.
  • kk is the spring constant, indicating how stiff the spring is.

This equation is essential for understanding our system's physics.

The Harmonic Solution: It's Beautiful

Now, here's the cool part. When conditions are right (meaning the damping isn't too strong), our system behaves in a super cool way. We can describe its motion with a beautiful equation:

x(t)=Aeγtcos(ωt+ϕ)+Cx(t) = A e^{-\gamma t} \cos(\omega t + \phi) + C

Let me break it down:

  • AA is the amplitude, which is like how far our mass swings.
  • γ\gamma represents damping coefficient, which resists motion.
  • ω\omega is the angular frequency.
  • ϕ\phi is the phase angle.
  • CC is just a constant that might shift our motion up or down.

Relating to a Damped Pendulum

You might wonder, how is this related to a damped pendulum? Well, it turns out, the equations are pretty similar! In the case of a damped pendulum, instead of a mass moving along a spring, we have a bob swinging on a string.

The equation for a damped pendulum looks like this:

d2θ(t)dt2+2γdθ(t)dt+ω02θ(t)=0\frac{d^2\theta(t)}{dt^2} + 2\gamma \frac{d\theta(t)}{dt} + \omega_0^2\theta(t) = 0
  • θ(t)\theta(t) is the angle our pendulum makes with the vertical.
  • γ\gamma represents damping coefficient, which resists motion.
  • ω0\omega_0 is the natural angular frequency of the pendulum, and it depends on the length of the string and gravity.

Why ω\omega Matters

In both systems, the value of ω\omega is super important! It tells us how fast things are moving or swinging. For a pendulum, it's linked to the length of the string. It's like the heartbeat of our system, determining how it behaves over time.

Measuring Pendulum String Length

Let's explore a clever method for measuring the length (LL) of a pendulum string, taking into account the damped conditions of the pendulum's motion. This approach is not only ingenious but also practical, especially when dealing with complex pendulum systems.

Step 1: Analyzing Damped Harmonic Motion

Our first step involves carefully observing the pendulum's harmonic motion under damped conditions. We need to record essential data, with a keen focus on determining the angular frequency (ω\omega) that characterizes the pendulum's oscillation in the presence of damping.

Step 2: The Damped ω\omega and LL Relationship

The angular frequency (ωω) for a damped pendulum system can be expressed in terms of the natural angular frequency (ω0ω_0) and the damping coefficient (γγ) as follows:

ω=ω02γ2ω = \sqrt{ω_0^2 - γ^2}

Where:

  • ωω is the angular frequency of the damped pendulum.
  • ω0ω_0 is the natural angular frequency (undamped) of the pendulum, given by ω0=gLω_0 = \sqrt{\frac{g}{L}}, where gg is the acceleration due to gravity and LL is the length of the pendulum.
  • γγ (gamma) is the damping coefficient, representing the strength of damping in the system.

This formula accounts for the effect of damping on the oscillatory motion of the pendulum. When γγ is non-zero, it reduces the angular frequency ωω from its undamped value ω0ω_0, resulting in damped oscillations.

The formula for the natural angular frequency (ω0\omega_0) of an undamped pendulum is given by:

ω0=gL\omega_0 = \sqrt{\frac{g}{L}}

Where:

  • ω0\omega_0 is the natural angular frequency of the undamped pendulum.
  • gg is the acceleration due to gravity.
  • LL is the length of the pendulum.

This equation tells us that the natural angular frequency (ω0\omega_0) is inversely proportional to the square root of the length of the pendulum (LL). Therefore, longer pendulums have smaller natural angular frequencies, and shorter pendulums have larger natural angular frequencies.

Step 3: Determining LL

We can rearrange the equation to calculate the length of the pendulum string (LL):

L=gω02L = \frac{g}{\omega_0^2}

Step 4: Handling Compound Pendulums

A compound pendulum, also known as a physical pendulum, is different from a simple pendulum because it has an extended mass distribution rather than a point mass at the end of a string. To calculate the angular frequency (ω0\omega_0) for a compound pendulum, you need to consider the moment of inertia (MOI) of the pendulum about its pivot point.

The moment of inertia (II) depends on the shape and distribution of mass in the pendulum. The moment of inertia of the spherical bob about its center is given by:

Ibob=25mr2I_{\text{bob}} = \frac{2}{5} m r^2

Where:

  • mm is the mass of the spherical bob.
  • rr is the radius of the spherical bob.

The moment of inertia of the pendulum about its pivot point, taking into account both the bob and the string, is the sum of the moments of inertia of the bob and the string:

I=Ibob+ml2I = I_{\text{bob}} + ml^2

Where:

  • II is the moment of inertia of the compound pendulum about its pivot point.
  • ml2ml^2 represents the moment of inertia of the string of length ll.

Now, you can use the formula for the natural frequency (ω0\omega_0) of the compound pendulum:

ω0=mgdI\omega_0 = \sqrt{\frac{mgd}{I}}

Where:

  • ω0\omega_0 is the natural frequency of the compound pendulum.
  • mm is the mass of the pendulum.
  • gg is the acceleration due to gravity.
  • dd is the distance between the pivot point and the center of mass of the pendulum.
  • II is the moment of inertia of the pendulum about the pivot point.

So, we can say ω0\omega_0 (natural frequency of a damped pendulum) is a function of both length (LL) and radius (rr).

We have reached a point where we have gathered essential information to grasp the project. By utilizing the curve_fit, I can extract the suitable parameters for the model. Subsequently, I apply these parameters to the physics principles discussed earlier to calculate the length of the pendulum string. However, determining the radius of the bob is more intricate, as the angular frequency (ω\omega) relies on both the length and radius. To address this challenge, I derive the length of the string using the assumption that the natural angular frequency (ω0\omega_0) adheres to the formula ω0=gl\omega_0 = \sqrt{\frac{g}{l}}, where gg is the acceleration due to gravity.


Until our next coding escapade, happy coding, and may your bugs be as elusive as Schroedinger's cat! 🐱💻✨