User Tools

Site Tools


visual3d:tutorials:emg:typical_emg_processing

Differences

This shows you the differences between two versions of the page.

Link to this comparison view

Both sides previous revisionPrevious revision
Next revision
Previous revision
visual3d:tutorials:emg:typical_emg_processing [2024/06/19 12:54] sgrangervisual3d:tutorials:emg:typical_emg_processing [2024/11/15 18:04] (current) – [Open Files and Tag] wikisysop
Line 1: Line 1:
-====== introduction ======+====== Typical EMG Processing ======
  
-**emg** stands for **electromyography**, which is the study of electrical signals from active muscles that are receiving input from the central nervous system. more information on emg can be found in most good biomechanics and motor control textbooks, and on [[http://en.wikipedia.org/wiki/electromyography|wikipedia]]. additional information on emg processing requirements for the [[http://www.isek-online.org/standards.html|international society of electrophysiology and kinesiology]]. you will also find other useful information on emg signal processing posted by [[[http://noraxon.com/downloads/educational.php3|noraxon]]]+**EMG** stands for **electromyography**, which is the study of electrical signals from active muscles that are receiving input from the central nervous system. This wiki includes an overview of [[visual3d:documentation:emg:emg_overview|how to process EMG data in Visual3D]]. More information on EMG can be found in most good biomechanics and motor control textbooks, and on [[http://en.wikipedia.org/wiki/Electromyography|Wikipedia]]. Additional information on EMG processing requirements for the [[http://www.isek-online.org/standards.html|International Society of Electrophysiology and Kinesiology]]. You will also find other useful information on EMG signal processing posted by [[http://noraxon.com/downloads/educational.php3|Noraxon]].
  
-this tutorial will show the basic steps for processing emg data into visual3dfiles used in this tutorial can be found here: [[[https://www.has-motion.com/download/examples/emg_tutorial.zip|emg tutorial]]]+===== Overview ===== 
 +This tutorial will show the basic steps for processing EMG data into Visual3DFiles used in this tutorial can be found here: [[https://www.has-motion.com/download/examples/EMG_Tutorial.zip|EMG Tutorial]]
  
-the tutorial will go through a step by step process using visual 3d pipeline commands to process the example emg data. it will do the following:+===== Visual 3D Implementation =====
  
-  - open files and tags +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.
-  - create pipeline parameter "emg signals+
-  - apply a band pass filter +
-  - compute an envelope +
-  - normalize based off of 4 different methods+
  
-====== visual 3d implementation ======+All of Visual3D's [[visual3d:documentation:pipeline:signal_commands:signal_commands_overview|pipeline commands for signal processing]] can be used with EMG data.
  
-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.+===== Processing EMG Signals with the Visual3D Pipeline =====
  
-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.+The tutorial will go through a step by step process using [[visual3d:documentation:pipeline:general_information:pipeline_commands_reference|Visual3D pipeline commands]] to process the example EMG dataIt will do the following:
  
-all signal processing commands in the pipeline can be used with emg data.+  - Open Files and Tags 
 +  - Create Pipeline Parameter "EMG Signals" 
 +  - Apply a Band Pass Filter 
 +  - Compute an Envelope 
 +  - Normalize based off of 4 different methods
  
-====== emg processing using visual 3d pipeline commands ======+==== Open Files and Tag ====
  
-the tutorial will go through a step by step process using visual 3d pipeline commands to process the example emg data. +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 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.+
  
 <code> <code>
 !-------------------------------------------------------------------------------------- !--------------------------------------------------------------------------------------
-open files and automatically assign tag to motion files for movement and mvic trials+Open files and automatically assign tag to motion files for Movement and MVIC trials
 ! -------------------------------------------------------------------------------------- ! --------------------------------------------------------------------------------------
    
-fresh workspace +Fresh Workspace 
-file_new+File_New
 ; ;
  
-!open files +!Open files 
-file_open +File_Open 
-! /file_name=+! /FILE_NAME=
 ; ;
  
-!assign tags to mvc_a file +!Assign Tags to MVC_A file 
-assign_tags_to_file +Assign_Tags_To_File 
-/motion_file_names=*mvc_a.c3d +/MOTION_FILE_NAMES=*MVC_A.C3D 
-! /query+! /QUERY
-/tags=mvc_a+/TAGS=MVC_A
 ; ;
  
-!assign tags to mvc_b file +!Assign Tags to MVC_B file 
-assign_tags_to_file +Assign_Tags_To_File 
-/motion_file_names=*mvc_b.c3d +/MOTION_FILE_NAMES=*MVC_B.C3D 
-! /query+! /QUERY
-/tags=mvc_b+/TAGS=MVC_B
 ; ;
  
-!assign tags to mvc_c file +!Assign Tags to MVC_C file 
-assign_tags_to_file +Assign_Tags_To_File 
-/motion_file_names=*mvc_c.c3d +/MOTION_FILE_NAMES=*MVC_C.C3D 
-! /query+! /QUERY
-/tags=mvc_c +/TAGS=MVC_C 
-';+;
  
-!assign tags to walking files +!Assign Tags to walking files 
-assign_tags_to_file +Assign_Tags_To_File 
-/motion_file_names=*trial*.c3d +/MOTION_FILE_NAMES=*trial*.C3D 
-!/query+!/QUERY
-/tags=movement+/TAGS=MOVEMENT
 ; ;
 </code> </code>
  
-===== create pipeline parameter "emg signals=====+==== Create Pipeline Parameter "EMG Signals" ====
  
-use the command **set_pipeline_parameter**, to set up a pipeline parameter to contain the emg signal names.+Use the command **Set_Pipeline_Parameter**, to set up a pipeline parameter to contain the EMG signal names.
  
 <code> <code>
-set_pipeline_parameter +Set_Pipeline_Parameter 
-/parameter_name=emg_signals +/PARAMETER_NAME=EMG_SIGNALS 
-/parameter_value=emg_a+emg_b+emg_c+/PARAMETER_VALUE=EMG_A+EMG_B+EMG_C
 ; ;
 </code> </code>
  
-===== apply band pass filter =====+==== Apply 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.+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 hzthe purpose of this high pass filter is to remove movement artifact (low frequency content).+**Note:** that the highpass filter often has a cutoff frequency anywhere between 20 and 50 HzThe purpose of this high pass filter is to remove movement artifact (low frequency content).
  
 <code> <code>
 !------------------------------------------------------------- !-------------------------------------------------------------
-apply a band pass filter - highpass and lowpass filter+Apply a band pass filter - highpass and lowpass filter
 !-------------------------------------------------------------  !------------------------------------------------------------- 
  
-select active file all_files +Select active file ALL_FILES 
-select_active_file  +Select_Active_File  
-/file_name=all_files  +/FILE_NAME=ALL_FILES  
-! /query+! /QUERY
 ; ;
  
-apply highpass filter with 50 hz cutoff +Apply highpass filter with 50 Hz cutoff 
-highpass_filter  +Highpass_Filter  
-/signal_types=analog  +/Signal_Types=ANALOG  
-/signal_names=::emg_signals  +/SIgnal_Names=::EMG_SIGNALS  
-/signal_folder=original  +/Signal_Folder=ORIGINAL  
-! /result_suffix=  +! /Result_Suffix=  
-! /result_folder=processed  +! /Result_Folder=PROCESSED  
-! /filter_class=butterworth  +! /Filter_Class=BUTTERWORTH  
-/frequency_cutoff=50  +/Frequency_Cutoff=50  
-/num_reflected= 0  +/Num_Reflected= 0  
-/total_buffer_size=100  +/Total_Buffer_Size=100  
-/num_bidirectional_passes=1 +/Num_Bidirectional_Passes=1 
  
  
-apply lowpass filter with 500 hz cutoff +Apply lowpass filter with 500 Hz cutoff 
-lowpass_filter  +Lowpass_Filter  
-/signal_types=analog  +/Signal_Types=ANALOG  
-/signal_names=::emg_signals  +/SIgnal_Names=::EMG_SIGNALS  
-/signal_folder=processed  +/Signal_Folder=PROCESSED  
-! /result_suffix=  +! /Result_Suffix=  
-! /result_folder=processed  +! /Result_Folder=PROCESSED  
-! /filter_class=butterworth  +! /Filter_Class=BUTTERWORTH  
-/frequency_cutoff=500  +/Frequency_Cutoff=500  
-/num_reflected= 0  +/Num_Reflected= 0  
-/total_buffer_size=100  +/Total_Buffer_Size=100  
-/num_bidirectional_passes=1 +/Num_Bidirectional_Passes=1 
 ; ;
 </code> </code>
  
-===== compute an envelope =====+==== 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:documentation:pipeline:signal_commands:moving_rms|moving rms]] page.+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:Documentation:Pipeline:Signal_Commands:Moving_RMS|Moving RMS]] page.
  
-the command below will apply a moving rms with a 100ms window+The command below will apply a moving RMS with a 100ms window
  
 <code> <code>
 !------------------------------------------- !-------------------------------------------
-apply a moving rms with a 100ms window+Apply a moving RMS with a 100ms window
 !------------------------------------------- !-------------------------------------------
-moving_rms  +Moving_RMS  
-/signal_types=analog  +/SIGNAL_TYPES=ANALOG  
-/signal_names=::emg_signals  +/SIGNAL_NAMES=::EMG_SIGNALS  
-/signal_folder=processed  +/SIGNAL_FOLDER=PROCESSED  
-! /result_suffix=  +! /RESULT_SUFFIX=  
-/result_folder=rms  +/RESULT_FOLDER=RMS  
-/num_window_frames=1+0.1*parameter::analog::rate +/NUM_WINDOW_FRAMES=1+0.1*PARAMETER::ANALOG::RATE 
 ; ;
 </code> </code>
  
-===== create normalization factor and scale =====+==== Create Normalization Factor and Scale ====
  
-there are four alternative normalization methods describes in this tutorial.+There are four alternative normalization methods describes in this tutorial.
  
-  - [[visual3d:documentation:emg:filtering:emg_envelope_version_3|normalize emg to mvc]] +  - [[Visual3D:Documentation:EMG:Filtering:EMG_Envelope_Version_3|Normalize EMG to MVC]] 
-  - [[visual3d:documentation:emg:processing:normalize_emg_to_generic_global_variable|normalize emg to generic global variable]] +  - [[Visual3D:Documentation:EMG:Processing:Normalize_EMG_to_Generic_Global_Variable|Normalize EMG to Generic Global Variable]] 
-  - [[visual3d:documentation:emg:filtering:removing_dc_bias|normalize to maximum 30 second interval]] +  - [[Visual3D:Documentation:EMG:Filtering:Removing_DC_Bias|Normalize to Maximum 30 second interval]] 
-  - [[visual3d:documentation:emg:processing:normalize_emg_to_mvc|normalize to the maximum within each gait cycle]]+  - [[Visual3D:Documentation:EMG:Processing:Normalize_EMG_to_MVC|Normalize to the Maximum within each Gait Cycle]]
  
-the beginning of the pipeline for all methods is as follows.+The beginning of the pipeline for all methods is as follows.
  
 <code> <code>
 !-------------------------------------------------------------------------------------------------- !--------------------------------------------------------------------------------------------------
-there are often artifacts in the emg signal related to filtering, so we want to ignore the first +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+! and last 50 frames of data (at POINT rate) when we create the normalization (scale) factors
 !-------------------------------------------------------------------------------------------------- !--------------------------------------------------------------------------------------------------
  
-event_explicit +Event_Explicit 
-/event_name=start +/EVENT_NAME=START 
-/frame=50 +/FRAME=50 
-! /time=+! /TIME=
 ; ;
  
-event_explicit +Event_Explicit 
-/event_name=end +/EVENT_NAME=END 
-/frame=250 +/FRAME=250 
-! /time=+! /TIME=
 ; ;
  
 !--------------------------------------------------- !---------------------------------------------------
-set pipeline parameters for start and end events+Set Pipeline parameters for START and END events
 !--------------------------------------------------- !---------------------------------------------------
  
-set_pipeline_parameter  +Set_Pipeline_Parameter  
-/parameter_name=start_at_event  +/PARAMETER_NAME=START_AT_EVENT  
-/parameter_value=start +/PARAMETER_VALUE=START 
 ; ;
  
-set_pipeline_parameter  +Set_Pipeline_Parameter  
-/parameter_name=end_at_event   +/PARAMETER_NAME=END_AT_EVENT   
-/parameter_value=end +/PARAMETER_VALUE=END 
 ; ;
 </code> </code>
  
-the following for loop will loop through each of the signals and first create a global event for the maximum and mean.+The following for loop will loop through each of the signals and first create a global event for the maximum and mean.
  
 <code> <code>
 !--------------------------------------------------- !---------------------------------------------------
-loop to calculate scale to normalize emg signals+Loop to calculate scale to normalize EMG signals
 !--------------------------------------------------- !---------------------------------------------------
-for_each  +For_Each  
-/iteration_parameter_name=index  +/ITERATION_PARAMETER_NAME=INDEX  
-/items=::emg_signals +/ITEMS=::EMG_SIGNALS 
 ; ;
  
-create an event at the maximum value +Create an event at the maximum value 
-event_global_maximum  +Event_Global_Maximum  
-/signal_types=analog  +/SIGNAL_TYPES=ANALOG  
-/signal_names=::index  +/SIGNAL_NAMES=::INDEX  
-/signal_folder=rms  +/SIGNAL_FOLDER=RMS  
-/event_name=emg_max  +/EVENT_NAME=EMG_MAX  
-/select_x=true  +/SELECT_X=TRUE  
-! /select_y=false  +! /SELECT_Y=FALSE  
-! /select_z=false  +! /SELECT_Z=FALSE  
-/start_at_event=::start_at_event  +/START_AT_EVENT=::START_AT_EVENT  
-/end_at_event=::end_at_event +/END_AT_EVENT=::END_AT_EVENT 
 ; ;
  
-create an event mean_start 15 frames prior to the max event +Create an event MEAN_START 15 frames prior to the max event 
-event_copy  +Event_Copy  
-/event_name=emg_max  +/EVENT_NAME=EMG_MAX  
-! /event_instance=0  +! /EVENT_INSTANCE=0  
-! /start_at_event=  +! /START_AT_EVENT=  
-! /end_at_event=  +! /END_AT_EVENT=  
-/new_event_name=mean_start  +/NEW_EVENT_NAME=MEAN_START  
-/frame_offset=-15 +/FRAME_OFFSET=-15 
 ; ;
  
-create an event mean_end 15 frames after the max event +Create an event MEAN_END 15 frames after the max event 
-event_copy  +Event_Copy  
-/event_name=emg_max  +/EVENT_NAME=EMG_MAX  
-! /event_instance=0  +! /EVENT_INSTANCE=0  
-! /start_at_event=  +! /START_AT_EVENT=  
-! /end_at_event=  +! /END_AT_EVENT=  
-/new_event_name=mean_end  +/NEW_EVENT_NAME=MEAN_END  
-/frame_offset=15 +/FRAME_OFFSET=15 
  
 </code> </code>
  
-the mean of the signal and the maximum of the signal will then be computed.+The mean of the signal and the maximum of the signal will then be computed.
  
 <code> <code>
-calculate mean of emg signal between mean_start and mean_end  +Calculate mean of EMG signal between MEAN_START and MEAN_END  
-metric_mean  +Metric_Mean  
-/result_metric_name=::index&_scale  +/RESULT_METRIC_NAME=::INDEX&_SCALE  
-! /apply_as_suffix_to_signal_name=false  +! /APPLY_AS_SUFFIX_TO_SIGNAL_NAME=FALSE  
-/result_metric_folder=emg_scale  +/RESULT_METRIC_FOLDER=EMG_SCALE  
-/signal_types=analog  +/SIGNAL_TYPES=ANALOG  
-/signal_names=::index  +/SIGNAL_NAMES=::INDEX  
-/signal_folder=rms  +/SIGNAL_FOLDER=RMS  
-/signal_components=0  +/SIGNAL_COMPONENTS=0  
-/event_sequence=mean_start+mean_end  +/EVENT_SEQUENCE=MEAN_START+MEAN_END  
-/exclude_events=  +/EXCLUDE_EVENTS=  
-/generate_mean_and_stddev=false  +/GENERATE_MEAN_AND_STDDEV=FALSE  
-! /append_to_existing_values=false +! /APPEND_TO_EXISTING_VALUES=FALSE 
 ; ;
  
 ! calculate the global maximum of the mean signal ! calculate the global maximum of the mean signal
-metric_maximum  +Metric_Maximum  
-/result_metric_name=_max  +/RESULT_METRIC_NAME=_MAX  
-/apply_as_suffix_to_signal_name=true  +/APPLY_AS_SUFFIX_TO_SIGNAL_NAME=TRUE  
-/result_metric_folder=processed  +/RESULT_METRIC_FOLDER=PROCESSED  
-/signal_types=metric  +/SIGNAL_TYPES=METRIC  
-/signal_names=::index&_scale  +/SIGNAL_NAMES=::INDEX&_SCALE  
-/signal_folder=emg_scale  +/SIGNAL_FOLDER=EMG_SCALE  
-! /signal_components=all_components  +! /SIGNAL_COMPONENTS=ALL_COMPONENTS  
-/event_sequence=  +/EVENT_SEQUENCE=  
-/exclude_events=  +/EXCLUDE_EVENTS=  
-! /generate_mean_and_stddev=true  +! /GENERATE_MEAN_AND_STDDEV=TRUE  
-! /append_to_existing_values=false  +! /APPEND_TO_EXISTING_VALUES=FALSE  
-/create_global_maximum=false +/CREATE_GLOBAL_MAXIMUM=FALSE 
 ; ;
  
-clean up some temporary signals  +Clean up some temporary signals  
-event_delete  +Event_Delete  
-/event_name=emg_max+mean_start+mean_end  +/EVENT_NAME=EMG_MAX+MEAN_START+MEAN_END  
-! /event_sequence=  +! /EVENT_SEQUENCE=  
-! /exclude_events=  +! /EXCLUDE_EVENTS=  
-! /time+! /TIME
 ; ;
  
-remove_signals  +Remove_Signals  
-/signal_types=metric  +/SIGNAL_TYPES=METRIC  
-/signal_names=::index&_scale_max  +/SIGNAL_NAMES=::INDEX&_SCALE_MAX  
-/signal_folder=processed +/SIGNAL_FOLDER=PROCESSED 
 ; ;
  
-end_for_each  +End_For_Each  
-/iteration_parameter_name=index +/ITERATION_PARAMETER_NAME=INDEX 
 ; ;
 </code> </code>
  
-==== normalize emg to mvc ====+==== 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).+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).
  
 <code> <code>
 ! -------------------------------------------------------------  ! ------------------------------------------------------------- 
-!   normalize+!   Normalize
 ! ------------------------------------------------------------- ! -------------------------------------------------------------
  
-for each emg signal, divide by it's maximum value+For each EMG signal, divide by it's maximum value
  
-for_each +For_Each 
-/iteration_parameter_name=current_signal +/Iteration_Parameter_Name=CURRENT_SIGNAL 
-/items=emg_a+emg_b+emg_c+/Items=EMG_A+EMG_B+EMG_C
 ; ;
  
- ! divide the emg signal by it's maximum value, which was stored automatically in the global workspace + ! Divide the EMG signal by it's maximum value, which was stored automatically in the GLOBAL workspace 
  
- evaluate_expression + Evaluate_Expression 
- /expressionanalog::original::&::current_signal& / global::metric::mvc_max::&::current_signal&_max + /EXPRESSIONANALOG::ORIGINAL::&::CURRENT_SIGNAL& / GLOBAL::METRIC::MVC_MAX::&::CURRENT_SIGNAL&_MAX 
- ! /signal_types+ ! /SIGNAL_TYPES
- ! /signal_folder=original + ! /SIGNAL_FOLDER=ORIGINAL 
- ! /signal_names+ ! /SIGNAL_NAMES
- /result_types=analog + /RESULT_TYPES=ANALOG 
- /result_folders=normalized + /RESULT_FOLDERS=NORMALIZED 
- /result_name=::current_signal + /RESULT_NAME=::CURRENT_SIGNAL 
- ! /apply_as_suffix_to_signal_name=false+ ! /APPLY_AS_SUFFIX_TO_SIGNAL_NAME=FALSE
  ;  ;
  
-end_for_each  +End_For_Each  
-/iteration_parameter_namecurrent_signal+/Iteration_Parameter_NameCURRENT_SIGNAL
 ; ;
 </code> </code>
  
-==== normalize emg to generic global variable ====+==== 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 workspacethis pipeline normalizes the emg signal to a 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 WorkspaceThis pipeline normalizes the EMG signal to a global variable.
  
 <code> <code>
 ! -------------------------------------------------------------  ! ------------------------------------------------------------- 
-!   normalize+!   Normalize
 ! ------------------------------------------------------------- ! -------------------------------------------------------------
  
-for each emg signal, divide by it's maximum value+For each EMG signal, divide by it's maximum value
  
-for_each +For_Each 
-/iteration_parameter_name=current_signal +/Iteration_Parameter_Name=CURRENT_SIGNAL 
-/items=emg_a+emg_b+emg_c+/Items=EMG_A+EMG_B+EMG_C
 ; ;
  
- ! divide the emg signal by it's maximum value, which was stored in the global workspace + ! Divide the EMG signal by it's maximum value, which was stored in the GLOBAL workspace 
  
- evaluate_expression  + Evaluate_Expression  
- /expressionanalog::original::&::current_signal& / global::metric::mvc_max::&::current_signal&_max + /EXPRESSIONANALOG::ORIGINAL::&::CURRENT_SIGNAL& / GLOBAL::METRIC::MVC_MAX::&::CURRENT_SIGNAL&_MAX 
- ! /signal_types+ ! /SIGNAL_TYPES
- ! /signal_folder=original + ! /SIGNAL_FOLDER=ORIGINAL 
- ! /signal_names+ ! /SIGNAL_NAMES
- /result_types=analog + /RESULT_TYPES=ANALOG 
- /result_folders=normalized + /RESULT_FOLDERS=NORMALIZED 
- /result_name=::current_signal + /RESULT_NAME=::CURRENT_SIGNAL 
- ! /apply_as_suffix_to_signal_name=false+ ! /APPLY_AS_SUFFIX_TO_SIGNAL_NAME=FALSE
  ;  ;
  
-end_for_each  +End_For_Each  
-/iteration_parameter_namecurrent_signal+/Iteration_Parameter_NameCURRENT_SIGNAL
 ; ;
 </code> </code>
  
-==== normalize to maximum 30 second interval ====+==== Approach 3: Normalize to Maximum 30 second interval ====
  
-this pipeline normalizes to the greatest ½ second activity during a 5 second trial. 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 emgthe average integrated emg during the ½ second is used to compute the normalization factor. it does the following:+This pipeline normalizes to the greatest ½ second activity during a 5 second trial. 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 EMGThe average integrated EMG during the ½ second is used to compute the normalization factor. It does the following:
  
-  * compute a normalization factor +  * 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) +  * Normalize to greatest ½ second during 5 second trial. This is accomplished by using an average filter (see the bottom of the page) 
-  * 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+  * 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.+  * The average integrated EMG during the ½ second is used to compute the normalization factor.
  
 <code> <code>
 ! ---------------------------------------------------------------  ! --------------------------------------------------------------- 
-!   normalize+!   Normalize
 ! ---------------------------------------------------------------  ! --------------------------------------------------------------- 
-for_each  +For_Each  
-/iteration_parameter_name=index  +/ITERATION_PARAMETER_NAME=INDEX  
-/items=::emg_signals +/ITEMS=::EMG_SIGNALS 
 ; ;
  
-evaluate_expression  +Evaluate_Expression  
-/expression=analog::rms&:&:&::index&/&global::metric::processed&:&:&::index&_scale_max_mean  +/EXPRESSION=ANALOG::RMS&:&:&::INDEX&/&GLOBAL::METRIC::PROCESSED&:&:&::INDEX&_SCALE_MAX_MEAN  
-/result_name=::index  +/RESULT_NAME=::INDEX  
-/result_type=analog  +/RESULT_TYPE=ANALOG  
-/result_folder=result +/RESULT_FOLDER=RESULT 
 ; ;
  
-end_for_each  +End_For_Each  
-/iteration_parameter_name=index +/ITERATION_PARAMETER_NAME=INDEX 
 ; ;
 </code> </code>
  
-==== normalize to the maximum within each gait cycle ====+==== Approach 4: Normalize to the Maximum within Each Gait Cycle ====
  
-for this example, the goal is as follows:+For this example, the goal is as follows:
  
-  * compute the maximum of the emg signal during each gait cycle +  * Compute the maximum of the EMG signal during each gait cycle 
-  * normalize the emg signal to the maximum 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.+The beginning of this pipeline are separating the left and right legs and identifying specific gait events depending on the leg.
  
 <code> <code>
-set_pipeline_parameter +Set_Pipeline_Parameter 
-/parameter_name=r_emg_signal_list +/PARAMETER_NAME=R_EMG_SIGNAL_LIST 
-/parameter_value=emg_a+emg_b+emg_c +/PARAMETER_VALUE=EMG_A+EMG_B+EMG_C 
-! /parameter_value_search_for+! /PARAMETER_VALUE_SEARCH_FOR
-! /parameter_value_replace_with+! /PARAMETER_VALUE_REPLACE_WITH
-! /parameter_value_prefix+! /PARAMETER_VALUE_PREFIX
-! /parameter_value_append+! /PARAMETER_VALUE_APPEND
-! /multi_pass=false+! /MULTI_PASS=FALSE
 ; ;
  
-set_pipeline_parameter +Set_Pipeline_Parameter 
-/parameter_name=l_emg_signal_list +/PARAMETER_NAME=L_EMG_SIGNAL_LIST 
-/parameter_value=emg_a+emg_b+emg_c +/PARAMETER_VALUE=EMG_A+EMG_B+EMG_C 
-! /parameter_value_search_for+! /PARAMETER_VALUE_SEARCH_FOR
-! /parameter_value_replace_with+! /PARAMETER_VALUE_REPLACE_WITH
-! /parameter_value_prefix+! /PARAMETER_VALUE_PREFIX
-! /parameter_value_append+! /PARAMETER_VALUE_APPEND
-! /multi_pass=false+! /MULTI_PASS=FALSE
 ; ;
  
-select_active_file +Select_Active_File 
-/file_name=all_files +/FILE_NAME=ALL_FILES 
-! /query=+! /QUERY=
 ; ;
    
-event_define_event_sequence +Event_Define_Event_Sequence 
-/event_sequence_name=rcycle_sequence +/EVENT_SEQUENCE_NAME=RCYCLE_SEQUENCE 
-/event_sequence=rhs+rto +/EVENT_SEQUENCE=RHS+RTO 
-! /exclude_events+! /EXCLUDE_EVENTS
-! /inside_of_sequence+! /INSIDE_OF_SEQUENCE
-! /offset_from_start+! /OFFSET_FROM_START
-! /offset_from_end+! /OFFSET_FROM_END
-! /offset_by=percent +! /OFFSET_BY=PERCENT 
-/event_sequence_instance=0+/EVENT_SEQUENCE_INSTANCE=0
 ; ;
  
-event_define_event_sequence +Event_Define_Event_Sequence 
-/event_sequence_name=lcycle_sequence +/EVENT_SEQUENCE_NAME=LCYCLE_SEQUENCE 
-/event_sequence=lhs+lto +/EVENT_SEQUENCE=LHS+LTO 
-! /exclude_events+! /EXCLUDE_EVENTS
-! /inside_of_sequence+! /INSIDE_OF_SEQUENCE
-! /offset_from_start+! /OFFSET_FROM_START
-! /offset_from_end+! /OFFSET_FROM_END
-! /offset_by=percent +! /OFFSET_BY=PERCENT 
-/event_sequence_instance=0+/EVENT_SEQUENCE_INSTANCE=0
  
 </code> </code>
  
-for loop will be used to loop through both the right and left leg.+for loop will be used to loop through both the right and left leg.
  
 <code> <code>
-for_each +For_Each 
-/iteration_parameter_name=side +/ITERATION_PARAMETER_NAME=SIDE 
-! /iteration_parameter_count_name+! /ITERATION_PARAMETER_COUNT_NAME
-/items=r+l+/ITEMS=R+L
 ; ;
  
-set_pipeline_parameter +Set_Pipeline_Parameter 
-/parameter_name=emg_signal_list +/PARAMETER_NAME=EMG_SIGNAL_LIST 
-/parameter_value=:&:&::side&_emg_signal_list +/PARAMETER_VALUE=:&:&::SIDE&_EMG_SIGNAL_LIST 
-! /parameter_value_search_for+! /PARAMETER_VALUE_SEARCH_FOR
-! /parameter_value_replace_with+! /PARAMETER_VALUE_REPLACE_WITH
-! /parameter_value_prefix+! /PARAMETER_VALUE_PREFIX
-! /parameter_value_append+! /PARAMETER_VALUE_APPEND
-/multi_pass=true+/MULTI_PASS=TRUE
 ; ;
 </code> </code>
  
-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.+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.
  
 <code> <code>
 ! --------------------------------------------------------------------- ! ---------------------------------------------------------------------
-get the maximum number of cycles in a trial the entire cmz file+Get the maximum number of cycles in a trial the entire CMZ file
 ! --------------------------------------------------------------------- ! ---------------------------------------------------------------------
  
  
-count the number of cycles in each trial+Count the number of cycles in each trial
  
-metric_event_sequence_count +Metric_Event_Sequence_Count 
-/result_metric_folder=emg_norm +/RESULT_METRIC_FOLDER=EMG_NORM 
-/result_metric_name=&::side&cycle_count +/RESULT_METRIC_NAME=&::SIDE&Cycle_Count 
-! /time_interval+! /TIME_INTERVAL
-/event_sequence=&::side&cycle_sequence +/EVENT_SEQUENCE=&::SIDE&CYCLE_SEQUENCE 
-/exclude_events+/EXCLUDE_EVENTS
-/generate_count_total_in_global=false+/GENERATE_COUNT_TOTAL_IN_GLOBAL=FALSE
 ; ;
  
-calculate the maximum number of cycles across all trials+Calculate the maximum number of cycles across all trials
    
-metric_maximum +Metric_Maximum 
-/result_metric_folder=emg_norm +/RESULT_METRIC_FOLDER=EMG_NORM 
-/result_metric_name=&::side&cycle_count_global +/RESULT_METRIC_NAME=&::SIDE&Cycle_Count_Global 
-! /apply_as_suffix_to_signal_name=false +! /APPLY_AS_SUFFIX_TO_SIGNAL_NAME=FALSE 
-/signal_types=metric +/SIGNAL_TYPES=METRIC 
-/signal_folder=emg_norm +/SIGNAL_FOLDER=EMG_NORM 
-/signal_names=&::side&cycle_count +/SIGNAL_NAMES=&::SIDE&Cycle_Count 
-/component_sequence=all +/COMPONENT_SEQUENCE=ALL 
-/event_sequence+/EVENT_SEQUENCE
-/exclude_events+/EXCLUDE_EVENTS
-/sequence_percent_start+/SEQUENCE_PERCENT_START
-/sequence_percent_end+/SEQUENCE_PERCENT_END
-/generate_mean_and_stddev=false +/GENERATE_MEAN_AND_STDDEV=FALSE 
-! /append_to_existing_values=false +! /APPEND_TO_EXISTING_VALUES=FALSE 
-/create_global_maximum=true+/CREATE_GLOBAL_MAXIMUM=TRUE
 ; ;
    
-create a list from 1 to the maximum event sequence in steps of 1+Create a list from 1 to the maximum event sequence in steps of 1
  
-set_pipeline_parameter_from_for_loop +Set_Pipeline_Parameter_From_For_Loop 
-/parameter_name=sequence_count +/PARAMETER_NAME=SEQUENCE_COUNT 
-/parameter_index_start=1 +/PARAMETER_INDEX_START=1 
-/parameter_index_end=global::metric::emg_norm::&::side&cycle_count_global_max +/PARAMETER_INDEX_END=GLOBAL::METRIC::EMG_NORM::&::SIDE&Cycle_Count_Global_Max 
-/parameter_index_step=1 +/PARAMETER_INDEX_STEP=1 
-/parameter_index_type=integer+/PARAMETER_INDEX_TYPE=INTEGER
 ; ;
 </code> </code>
  
-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.+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.
  
 <code> <code>
 ! --------------------------------------------------------------------- ! ---------------------------------------------------------------------
-create normalize signal+Create Normalize Signal
 ! --------------------------------------------------------------------- ! ---------------------------------------------------------------------
  
-create a temporary signal which contains zeros at all frames+Create a temporary signal which contains zeros at all frames
  
-evaluate_expression +Evaluate_Expression 
-/expression=current_signal * 0 +/EXPRESSION=CURRENT_SIGNAL * 0 
-/signal_types=analog +/SIGNAL_TYPES=ANALOG 
-/signal_folder=envelope +/SIGNAL_FOLDER=ENVELOPE 
-/signal_names=::emg_signal_list +/SIGNAL_NAMES=::EMG_SIGNAL_LIST 
-/result_types=derived +/RESULT_TYPES=DERIVED 
-/result_folders=emg_norm_temp +/RESULT_FOLDERS=EMG_NORM_TEMP 
-/result_name+/RESULT_NAME
-/apply_as_suffix_to_signal_name=true+/APPLY_AS_SUFFIX_TO_SIGNAL_NAME=TRUE
 ; ;
  
-loop over every sequence+Loop over every sequence
  
-for_each +For_Each 
-/iteration_parameter_name=sequence_current +/ITERATION_PARAMETER_NAME=SEQUENCE_CURRENT 
-! /iteration_parameter_count_name+! /ITERATION_PARAMETER_COUNT_NAME
-/items=::sequence_count+/ITEMS=::SEQUENCE_COUNT
 ; ;
  
- ! define the "current" sequence as temp_cycle+ ! Define the "current" sequence as TEMP_CYCLE
  
- event_define_event_sequence + Event_Define_Event_Sequence 
- /event_sequence_name=temp_cycle + /EVENT_SEQUENCE_NAME=TEMP_CYCLE 
- /event_sequence=&::side&cycle_sequence + /EVENT_SEQUENCE=&::SIDE&CYCLE_SEQUENCE 
- ! /exclude_events+ ! /EXCLUDE_EVENTS
- ! /inside_of_sequence+ ! /INSIDE_OF_SEQUENCE
- ! /offset_from_start+ ! /OFFSET_FROM_START
- ! /offset_from_end+ ! /OFFSET_FROM_END
- ! /offset_by=percent + ! /OFFSET_BY=PERCENT 
- /event_sequence_instance=::sequence_current+ /EVENT_SEQUENCE_INSTANCE=::SEQUENCE_CURRENT
  ;  ;
      
- ! loop over every emg signal+ ! Loop over every EMG Signal
      
- for_each + For_Each 
- /iteration_parameter_name=emg_signal_current + /ITERATION_PARAMETER_NAME=EMG_SIGNAL_CURRENT 
- ! /iteration_parameter_count_name+ ! /ITERATION_PARAMETER_COUNT_NAME
- /items=::emg_signal_list+ /ITEMS=::EMG_SIGNAL_LIST
  ;  ;
        
-  ! for all frames during the current cycle,  +  ! For all frames during the current cycle,  
-  ! set the values to the maximum of the emg signal+  ! set the values to the maximum of the EMG signal
        
-  set_data_to_new_values +  Set_Data_To_New_Values 
-  /signal_types=derived +  /SIGNAL_TYPES=DERIVED 
-  /signal_folder=emg_norm_temp +  /SIGNAL_FOLDER=EMG_NORM_TEMP 
-  /signal_names=::emg_signal_current +  /SIGNAL_NAMES=::EMG_SIGNAL_CURRENT 
-  ! /signal_components=all_components +  ! /SIGNAL_COMPONENTS=ALL_COMPONENTS 
-  /result_types=derived +  /RESULT_TYPES=DERIVED 
-  /result_folders=emg_norm_temp +  /RESULT_FOLDERS=EMG_NORM_TEMP 
-  /result_suffix+  /RESULT_SUFFIX
-  /event_sequence=temp_cycle +  /EVENT_SEQUENCE=TEMP_CYCLE 
-  ! /exclude_events+  ! /EXCLUDE_EVENTS
-  ! /start_frame+  ! /START_FRAME
-  ! /end_frame+  ! /END_FRAME
-  ! /use_point_rate=true +  ! /USE_POINT_RATE=TRUE 
-  /replacement_values=metric_maximum(analog::envelope::&::emg_signal_current&,event_label::sequence::temp_cycle+  /REPLACEMENT_VALUES=METRIC_MAXIMUM(ANALOG::ENVELOPE::&::EMG_SIGNAL_CURRENT&,EVENT_LABEL::SEQUENCE::TEMP_CYCLE
-  ! /threshold_high=no_data +  ! /THRESHOLD_HIGH=NO_DATA 
-  ! /threshold_low=no_data +  ! /THRESHOLD_LOW=NO_DATA 
-  ! /set_data_at_single_event=false+  ! /SET_DATA_AT_SINGLE_EVENT=FALSE
   ;    ; 
        
- end_for_each  + End_For_Each  
- /iteration_parameter_name=emg_signal_current+ /ITERATION_PARAMETER_NAME=EMG_SIGNAL_CURRENT
  ;  ;
  
-end_for_each  +End_For_Each  
-/iteration_parameter_name=sequence_current+/ITERATION_PARAMETER_NAME=SEQUENCE_CURRENT
 ; ;
 </code> </code>
  
-we can then normalize the emg signal by dividing it by the temporary signal above.+We can then normalize the EMG signal by dividing it by the temporary signal above.
  
 <code> <code>
 ! --------------------------------------------------------------------- ! ---------------------------------------------------------------------
-normalize emg signal+Normalize EMG Signal
 ! --------------------------------------------------------------------- ! ---------------------------------------------------------------------
  
-divide the emg signals by the temporary signal above+Divide the EMG signals by the temporary signal above
  
-for_each +For_Each 
-/iteration_parameter_name=emg_signal_current +/ITERATION_PARAMETER_NAME=EMG_SIGNAL_CURRENT 
-! /iteration_parameter_count_name+! /ITERATION_PARAMETER_COUNT_NAME
-/items=::emg_signal_list+/ITEMS=::EMG_SIGNAL_LIST
 ; ;
  
- evaluate_expression + Evaluate_Expression 
- /expression=analog::envelope::&::emg_signal_current&  + /EXPRESSION=ANALOG::ENVELOPE::&::EMG_SIGNAL_CURRENT&  
- /derived::emg_norm_temp::&::emg_signal_current+ /DERIVED::EMG_NORM_TEMP::&::EMG_SIGNAL_CURRENT
- ! /signal_types+ ! /SIGNAL_TYPES
- ! /signal_folder+ ! /SIGNAL_FOLDER
- ! /signal_names+ ! /SIGNAL_NAMES
- /result_types=derived + /RESULT_TYPES=DERIVED 
- /result_folders=emg_norm + /RESULT_FOLDERS=EMG_NORM 
- /result_name=&::emg_signal_current+ /RESULT_NAME=&::EMG_SIGNAL_CURRENT
- ! /apply_as_suffix_to_signal_name=false+ ! /APPLY_AS_SUFFIX_TO_SIGNAL_NAME=FALSE
  ;  ;
  
-end_for_each  +End_For_Each  
-/iteration_parameter_name=emg_signal_current+/ITERATION_PARAMETER_NAME=EMG_SIGNAL_CURRENT
 ; ;
  
-end_for_each  +End_For_Each  
-/iteration_parameter_name=side+/ITERATION_PARAMETER_NAME=SIDE
 ; ;
 </code> </code>
visual3d/tutorials/emg/typical_emg_processing.1718801658.txt.gz · Last modified: 2024/06/19 12:54 by sgranger