visual3d:tutorials:emg:typical_emg_processing
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
visual3d:tutorials:emg:typical_emg_processing [2024/06/26 20:08] – removed sgranger | visual3d:tutorials:emg:typical_emg_processing [2024/11/15 18:04] (current) – [Open Files and Tag] wikisysop | ||
---|---|---|---|
Line 1: | Line 1: | ||
+ | ====== Typical EMG Processing ====== | ||
+ | |||
+ | **EMG** stands for **electromyography**, | ||
+ | |||
+ | ===== Overview ===== | ||
+ | This tutorial will show the basic steps for processing EMG data into Visual3D. Files used in this tutorial can be found here: [[https:// | ||
+ | |||
+ | ===== Visual 3D Implementation ===== | ||
+ | |||
+ | Visual3D expects the EMG signals to be stored in the C3D file as ANALOG data, not unlike force platform data. One point of importance is that EMG typically has very high frequency content, which means that it must be sampled at high data sampling rates. As of Visual3D Version 5.0, EMG signals are stored as integer multiples of the motion capture sampling rate (e.g. consistent with the common use of the C3D file format). One of the nuisances caused by this requirement is that force platform data must be sampled at a very high rate in order to match the EMG sampling rate. | ||
+ | |||
+ | All of Visual3D' | ||
+ | |||
+ | ===== Processing EMG Signals with the Visual3D Pipeline ===== | ||
+ | |||
+ | The tutorial will go through a step by step process using [[visual3d: | ||
+ | |||
+ | - Open Files and Tags | ||
+ | - Create Pipeline Parameter "EMG Signals" | ||
+ | - Apply a Band Pass Filter | ||
+ | - Compute an Envelope | ||
+ | - Normalize based off of 4 different methods | ||
+ | |||
+ | ==== Open Files and Tag ==== | ||
+ | |||
+ | This section will walk you through pipeline commands that will open files and automatically assign tags to motion files for **Movement** and **MVIC** trials. | ||
+ | |||
+ | < | ||
+ | !-------------------------------------------------------------------------------------- | ||
+ | ! Open files and automatically assign tag to motion files for Movement and MVIC trials | ||
+ | ! -------------------------------------------------------------------------------------- | ||
+ | |||
+ | ! Fresh Workspace | ||
+ | File_New | ||
+ | ; | ||
+ | |||
+ | !Open files | ||
+ | File_Open | ||
+ | ! /FILE_NAME= | ||
+ | ; | ||
+ | |||
+ | !Assign Tags to MVC_A file | ||
+ | Assign_Tags_To_File | ||
+ | / | ||
+ | ! /QUERY= | ||
+ | /TAGS=MVC_A | ||
+ | ; | ||
+ | |||
+ | !Assign Tags to MVC_B file | ||
+ | Assign_Tags_To_File | ||
+ | / | ||
+ | ! /QUERY= | ||
+ | /TAGS=MVC_B | ||
+ | ; | ||
+ | |||
+ | !Assign Tags to MVC_C file | ||
+ | Assign_Tags_To_File | ||
+ | / | ||
+ | ! /QUERY= | ||
+ | /TAGS=MVC_C | ||
+ | ; | ||
+ | |||
+ | !Assign Tags to walking files | ||
+ | Assign_Tags_To_File | ||
+ | / | ||
+ | !/QUERY= | ||
+ | / | ||
+ | ; | ||
+ | </ | ||
+ | |||
+ | ==== Create Pipeline Parameter "EMG Signals" | ||
+ | |||
+ | Use the command **Set_Pipeline_Parameter**, | ||
+ | |||
+ | < | ||
+ | Set_Pipeline_Parameter | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | </ | ||
+ | |||
+ | ==== Apply a Band Pass Filter ==== | ||
+ | |||
+ | Apply a band pass filter to the EMG data with the commands **Highpass_Filter** and **Lowpass_Filter**. Surface EMG signals have a frequency content between 20 and 500 Hz. | ||
+ | |||
+ | **Note:** that the highpass filter often has a cutoff frequency anywhere between 20 and 50 Hz. The purpose of this high pass filter is to remove movement artifact (low frequency content). | ||
+ | |||
+ | < | ||
+ | !------------------------------------------------------------- | ||
+ | ! Apply a band pass filter - highpass and lowpass filter | ||
+ | !------------------------------------------------------------- | ||
+ | |||
+ | ! Select active file ALL_FILES | ||
+ | Select_Active_File | ||
+ | / | ||
+ | ! / | ||
+ | ; | ||
+ | |||
+ | ! Apply highpass filter with 50 Hz cutoff | ||
+ | Highpass_Filter | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | ! Apply lowpass filter with 500 Hz cutoff | ||
+ | Lowpass_Filter | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | </ | ||
+ | |||
+ | ==== Compute an Envelope ==== | ||
+ | |||
+ | Compute the linear envelope of the EMG signal by calculating the RMS value using the command **Moving_RMS** for short successive time periods (the time period is described as a moving window). More information on calculating the RMS can be found at the [[Visual3D: | ||
+ | |||
+ | The command below will apply a moving RMS with a 100ms window | ||
+ | |||
+ | < | ||
+ | !------------------------------------------- | ||
+ | ! Apply a moving RMS with a 100ms window | ||
+ | !------------------------------------------- | ||
+ | Moving_RMS | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | </ | ||
+ | |||
+ | ==== Create Normalization Factor and Scale ==== | ||
+ | |||
+ | There are four alternative normalization methods describes in this tutorial. | ||
+ | |||
+ | - [[Visual3D: | ||
+ | - [[Visual3D: | ||
+ | - [[Visual3D: | ||
+ | - [[Visual3D: | ||
+ | |||
+ | The beginning of the pipeline for all methods is as follows. | ||
+ | |||
+ | < | ||
+ | !-------------------------------------------------------------------------------------------------- | ||
+ | ! There are often artifacts in the EMG signal related to filtering, so we want to ignore the first | ||
+ | ! and last 50 frames of data (at POINT rate) when we create the normalization (scale) factors | ||
+ | !-------------------------------------------------------------------------------------------------- | ||
+ | |||
+ | Event_Explicit | ||
+ | / | ||
+ | /FRAME=50 | ||
+ | ! /TIME= | ||
+ | ; | ||
+ | |||
+ | Event_Explicit | ||
+ | / | ||
+ | /FRAME=250 | ||
+ | ! /TIME= | ||
+ | ; | ||
+ | |||
+ | !--------------------------------------------------- | ||
+ | ! Set Pipeline parameters for START and END events | ||
+ | !--------------------------------------------------- | ||
+ | |||
+ | Set_Pipeline_Parameter | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | Set_Pipeline_Parameter | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | </ | ||
+ | |||
+ | The following for loop will loop through each of the signals and first create a global event for the maximum and mean. | ||
+ | |||
+ | < | ||
+ | !--------------------------------------------------- | ||
+ | ! Loop to calculate scale to normalize EMG signals | ||
+ | !--------------------------------------------------- | ||
+ | For_Each | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | ! Create an event at the maximum value | ||
+ | Event_Global_Maximum | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | ! Create an event MEAN_START 15 frames prior to the max event | ||
+ | Event_Copy | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | ! Create an event MEAN_END 15 frames after the max event | ||
+ | Event_Copy | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | </ | ||
+ | |||
+ | The mean of the signal and the maximum of the signal will then be computed. | ||
+ | |||
+ | < | ||
+ | ! Calculate mean of EMG signal between MEAN_START and MEAN_END | ||
+ | Metric_Mean | ||
+ | / | ||
+ | ! / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | ; | ||
+ | |||
+ | ! calculate the global maximum of the mean signal | ||
+ | Metric_Maximum | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | ! Clean up some temporary signals | ||
+ | Event_Delete | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | ; | ||
+ | |||
+ | Remove_Signals | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | End_For_Each | ||
+ | / | ||
+ | ; | ||
+ | </ | ||
+ | |||
+ | ==== Approach 1: Normalize EMG to MVC ==== | ||
+ | |||
+ | This pipeline normalizes the EMG signal by dividing the EMG signal by the maximum value of the EMG signal from a separate trial (e.g. a trial that elicits a maximum voluntary contraction (MVC) from the muscle). | ||
+ | |||
+ | < | ||
+ | ! ------------------------------------------------------------- | ||
+ | ! | ||
+ | ! ------------------------------------------------------------- | ||
+ | |||
+ | ! For each EMG signal, divide by it's maximum value | ||
+ | |||
+ | For_Each | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | ! Divide the EMG signal by it's maximum value, which was stored automatically in the GLOBAL workspace | ||
+ | |||
+ | | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | ; | ||
+ | |||
+ | End_For_Each | ||
+ | / | ||
+ | ; | ||
+ | </ | ||
+ | |||
+ | ==== Approach 2: Normalize EMG to Generic Global Variable ==== | ||
+ | |||
+ | Typically the scaling factors for EMG signals are the result of other data trials (such as MVIC trials). Individual files in Visual3D cannot access the data from other files, so the only way signals can be shared is through the Global Workspace. This pipeline normalizes the EMG signal to a global variable. | ||
+ | |||
+ | < | ||
+ | ! ------------------------------------------------------------- | ||
+ | ! | ||
+ | ! ------------------------------------------------------------- | ||
+ | |||
+ | ! For each EMG signal, divide by it's maximum value | ||
+ | |||
+ | For_Each | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | ! Divide the EMG signal by it's maximum value, which was stored in the GLOBAL workspace | ||
+ | |||
+ | | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | ; | ||
+ | |||
+ | End_For_Each | ||
+ | / | ||
+ | ; | ||
+ | </ | ||
+ | |||
+ | ==== Approach 3: Normalize to Maximum 30 second interval ==== | ||
+ | |||
+ | This pipeline normalizes to the greatest ½ second activity during a 5 second trial. A window of twenty-five .02 second intervals of integrated EMG is moved one interval at a time across the 5 seconds of data to find the greatest EMG. The average integrated EMG during the ½ second is used to compute the normalization factor. It does the following: | ||
+ | |||
+ | * Compute a normalization factor | ||
+ | * Normalize to greatest ½ second during 5 second trial. This is accomplished by using an average filter (see the bottom of the page) | ||
+ | * A window of twenty-five .02 second intervals of integrated EMG is moved one interval at a time across the 5 seconds of data to find the greatest EMG. | ||
+ | * The average integrated EMG during the ½ second is used to compute the normalization factor. | ||
+ | |||
+ | < | ||
+ | ! --------------------------------------------------------------- | ||
+ | ! | ||
+ | ! --------------------------------------------------------------- | ||
+ | For_Each | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | Evaluate_Expression | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | End_For_Each | ||
+ | / | ||
+ | ; | ||
+ | </ | ||
+ | |||
+ | ==== Approach 4: Normalize to the Maximum within Each Gait Cycle ==== | ||
+ | |||
+ | For this example, the goal is as follows: | ||
+ | |||
+ | * Compute the maximum of the EMG signal during each gait cycle | ||
+ | * Normalize the EMG signal to the maximum during each gait cycle | ||
+ | |||
+ | The beginning of this pipeline are separating the left and right legs and identifying specific gait events depending on the leg. | ||
+ | |||
+ | < | ||
+ | Set_Pipeline_Parameter | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | ; | ||
+ | |||
+ | Set_Pipeline_Parameter | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | ; | ||
+ | |||
+ | Select_Active_File | ||
+ | / | ||
+ | ! /QUERY= | ||
+ | ; | ||
+ | |||
+ | Event_Define_Event_Sequence | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | Event_Define_Event_Sequence | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | / | ||
+ | ; | ||
+ | </ | ||
+ | |||
+ | A for loop will be used to loop through both the right and left leg. | ||
+ | |||
+ | < | ||
+ | For_Each | ||
+ | / | ||
+ | ! / | ||
+ | /ITEMS=R+L | ||
+ | ; | ||
+ | |||
+ | Set_Pipeline_Parameter | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | / | ||
+ | ; | ||
+ | </ | ||
+ | |||
+ | We're going to then loop over every sequence in the trial, so we first calculate the maximum number of sequences in the longest trial. | ||
+ | |||
+ | < | ||
+ | ! --------------------------------------------------------------------- | ||
+ | ! Get the maximum number of cycles in a trial the entire CMZ file | ||
+ | ! --------------------------------------------------------------------- | ||
+ | |||
+ | |||
+ | ! Count the number of cycles in each trial | ||
+ | |||
+ | Metric_Event_Sequence_Count | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | ! Calculate the maximum number of cycles across all trials | ||
+ | |||
+ | Metric_Maximum | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | ! Create a list from 1 to the maximum event sequence in steps of 1 | ||
+ | |||
+ | Set_Pipeline_Parameter_From_For_Loop | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | </ | ||
+ | |||
+ | We will then create a temporary signal to normalize the EMG signal with, which will contain the maximum of the EMG Signal of the current cycle. | ||
+ | |||
+ | < | ||
+ | ! --------------------------------------------------------------------- | ||
+ | ! Create Normalize Signal | ||
+ | ! --------------------------------------------------------------------- | ||
+ | |||
+ | ! Create a temporary signal which contains zeros at all frames | ||
+ | |||
+ | Evaluate_Expression | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | ! Loop over every sequence | ||
+ | |||
+ | For_Each | ||
+ | / | ||
+ | ! / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | ! Define the " | ||
+ | |||
+ | | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | / | ||
+ | ; | ||
+ | | ||
+ | ! Loop over every EMG Signal | ||
+ | | ||
+ | | ||
+ | / | ||
+ | ! / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | ! For all frames during the current cycle, | ||
+ | ! set the values to the maximum of the EMG signal | ||
+ | |||
+ | Set_Data_To_New_Values | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! /END_FRAME= | ||
+ | ! / | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | | ||
+ | |||
+ | | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | End_For_Each | ||
+ | / | ||
+ | ; | ||
+ | </ | ||
+ | |||
+ | We can then normalize the EMG signal by dividing it by the temporary signal above. | ||
+ | |||
+ | < | ||
+ | ! --------------------------------------------------------------------- | ||
+ | ! Normalize EMG Signal | ||
+ | ! --------------------------------------------------------------------- | ||
+ | |||
+ | ! Divide the EMG signals by the temporary signal above | ||
+ | |||
+ | For_Each | ||
+ | / | ||
+ | ! / | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | ! / | ||
+ | ! / | ||
+ | / | ||
+ | / | ||
+ | / | ||
+ | ! / | ||
+ | ; | ||
+ | |||
+ | End_For_Each | ||
+ | / | ||
+ | ; | ||
+ | |||
+ | End_For_Each | ||
+ | / | ||
+ | ; | ||
+ | </ | ||
+ | |||
+ | |||
visual3d/tutorials/emg/typical_emg_processing.1719432491.txt.gz · Last modified: 2024/06/26 20:08 by sgranger