The CalculateTPLikePropsFromSweep() calculates square pulse properties
+from a given sweep. It retrieves the associated setting values for the sweep and
+returns the voltage difference, current difference and resistance for each
+head stage. The function only works for current clamp mode.
The goal of the algorithm is to determine the steady state resistance of a
+square pulse response. The available sweep data contains the excitation pulse as
+well as the response pulse. The excitation pulse is in units of current and the
+response is measured in voltage. The actual data consists of discrete points of
+digital to analog output (excitation) and analog to digital input (response).
+For simplicity this will be neglected in the following.
+
The square pulse DA data is used to find the leading and trailing edge of
+the pulse. Therefore a level of 10% of the amplitude of the square pulse is defined
+to find the location of signal crossing.
+
+\[level = minimum + (maximum - minimum) * 0.1\]
+
Using a edge finding algorithm the location of firstEdge and secondEdge
+is retrieved.
+
For determination of the base line level in front of the pulse a
+range is defined to average the points of the square pulse response. The full base
+line fraction in front of a square pulse starts after the onSetDelay and ends
+at firstEdge. The length of the range is defined as 10% of this distance. The
+reference point is the end of this range and it is set close to firstEdge.
For determination of the elevated level at the end of the active pulse a
+range is defined to average the points of the square pulse response. The active
+pulse starts after firstEdge and ends with secondEdge. The length of the
+range is defined as 10% of this distance. The reference point is the end of this
+range and it is set close to secondEdge.
The difference between elevated level and base line level is the voltage response
+for the steady state.
+
+\[\Delta U = elevatedlevel - baselinelevel\]
+
To get the steady state resistance the current from the excitation amplitude of
+the square pulse is required. It is retrieved the same way from the square pulse
+excitation data as the voltage response from the square pulse response data.
+The difference from the square pulse excitation data yields:
The sweep number is retrieved, that allows to get the setting for this sweep.
+Also the configuration wave for the sweep is retrieved, that is required to
+extract the actual ADC and DAC data from the sweep.
+
The totalOnsetDelay is independent of the head stage and the sum of the auto and
+user onset delay. The number of the ADC channel that are used per head stage
+is read to wave ADCs. The same is done for the number of the DAC channel to the wave DACs.
+The units per head stage are read from the saved settings to wave ADunit and
+wave DAunit. If a head stage was active is read to the wave wave statusHS.
The following is done for each head stage up to NUM_HEADSTAGES (default = 8):
+
+
If the head stage was not active, continue with next head stage.
+
if(!statusHS[i])
+continue
+endif
+
+
+
With the number of the DAC channel of this head stage the column in the sweep
+with the actual data read with AFH_GetDAQDataColumn() to DAcol. The
+same is done for the column with the AD data to ADcol. With the columns the
+actual data is read to wave DA and AD respectively with
+ExtractOneDimDataFromSweep(). The coordinate in points of
+totalOnsetDelay on the scale of DA is saved to onsetDelayPoint.
The scaled x coordinates of the full square pulse range including base line are
+defined from totalOnsetDelay to the end of the DA wave and saved in
+first and last.
+
A signal level is defined for finding the edges in the sent square pulse
+(DA channel). The level is 10 % from the difference of maximum - minimum of the
+DA data above the minimum level. With FindLevels up to two signal crossings
+at the level are searched between first and last in wave DA and
+their x position in points is saved to wave levels. The search runs from
+lower to higher x coordinates.
+An assertion checks if two (or more) signal crossings were found.
The first found location is saved to firstEdge and the second to
+secondEdge. By default DA contains a pulse so the linear interpolation
+between the points done by FindLevels results that firstEdge is found
+at last_baseline_point + 0.1 and secondEdge at last_pulse_point + 0.9. The values
+are truncated to integers to equal the last baseline point as well as the last
+pulse point.
For determination of the base line the range is defined as 10 % of the
+firstEdge location to totalOnsetDelay. The end point of the range is
+firstEdge-1. The baseline level is then defined as the average of all
+AD points in this range.
The elevated range (steady state) is defined by 10 % of the firstEdge
+location to secondEdge. The elevated level is then defined as the
+average of all AD points in this range.
The following tree describes the NWB layout version 1
+
# specifications:
+# - version : 1.0.5 beta
+# url : https://htmlpreview.github.io/?https://raw.githubusercontent.com/NeurodataWithoutBorders/specification_nwbn_1_0_x/master/version_1.0.5_beta/nwb_file_format_specification_1.0.5_beta.html#File_organization
+
+acquisition:
+timeseries:# empty if no acquired data is saved
+-data_XXXXX_ADY:
+stimulus_description :name of the stimset# custom entry
+data :1D dataset with attributes unit, conversion and resolution
+electrode_name :Name of the electrode headstage, more info in /general/intracellular_ephys/electrode_name
+gain :scaling factor
+num_samples :Number of rows in data
+starting_time :relative to /session_start_time with attributes rate and unit
+
+# For Voltage Clamp (Missing entries are mentioned in missing_fields):
+capacitance_fast :
+capacitance_slow :
+resistance_comp_bandwidth :
+resistance_comp_correction :
+resistance_comp_prediction :
+whole_cell_capacitance_comp :
+whole_cell_series_resistance_comp :
+
+# For Current Clamp (Missing entries are mentioned in missing_fields):
+bias_current :
+bridge_balance :
+capacitance_compensation :
+
+description :Unused
+source :Human readable description of the source of the data
+comment :User comment for the sweep
+missing_fields :Entries missing for voltage clamp/current clamp data
+ancestry :Class hierarchy defined by NWB spec, important members are
+CurrentClampSeries, IZeroClampSeries and VoltageClampSeries
+neurodata_type :TimeSeries
+
+stimulus:
+presentation:# empty if no acquired data is saved
+-data_XXXXX_DAY:# DA data as sent to the neuron, including delays, scaling, initial TP, etc.
+data :1D dataset
+electrode_name :Name of the electrode headstage, more info in /general/intracellular_ephys/electrode_name
+gain :
+num_samples :Number of rows in data
+starting_time :relative to /session_start_time with attributes rate and unit
+description :Unused
+source :Human readable description of the source of the data
+ancestry :Class hierarchy defined by NWB spec, important members are
+CurrentClampStimulusSeries and VoltageClampStimulusSeries
+neurodata_type :TimeSeries
+template:unused
+
+general:
+file_create_date :text array with UTC modification timestamps
+identifier :SHA256 hash, ensured to be unique
+nwb_version :NWB specification version
+session_description :unused
+session_start_time :ISO8601 timestamp in UTC timezone with sub-second precision defining when the recording session started
+generated_by:# custom entry
+Nx2 text data array describing the system which created the data. First column is the key, second the value.
+history:# custom entry
+Full Igor Pro history since measurement start (useful for debugging)
+
+# The following attributes are only available if explicitly set by the user:
+data_collection :
+experiment_description :
+experimenter :
+institution :
+lab :
+notes :
+pharmacology :
+protocol :
+related_publications :
+session_id :
+slices :
+stimulus :
+age :
+description :
+genotype :
+sex :
+species :
+subject_id :
+weight :
+surgery :
+virus :
+
+devices:# empty if no acquired data is saved
+-device_XXX:# Name of the DA_ephys device, something like "Harvard Bioscience ITC 18USB"
+intracellular_ephys:
+electrode_XXX:# XXX can be set by the user via writing into GetCellElectrodeNames()
+description :Holds the description of the electrode, something like "Headstage 1".
+device :Device used to record the data
+labnotebook:# custom entry
+-XXXX:# Name of the device
+numericalKeys :Numerical labnotebook
+numericalValues :Keys for numerical labnotebook
+textualKeys :Keys for textual labnotebook
+textualValues :Textual labnotebook
+testpulse:# custom entry
+-XXXX:# Name of the device
+TPStorage/TPStorage_X:testpulse property waves
+StoredTestPulses:Raw testpulses as 1D datasets
+user_comment:
+-XXXX:# Name of the device
+userComment:All user comments from this session
+stimsets:# custom entry
+XXXXXX_[DA/TTL]_Y_[SegWvType/WP/WPT]:The Wavebuilder parameter waves. These waves will not be available for
+"third party stimsets" created outside of MIES.
+XXXXXX_[DA/TTL]_Y:Name of the stimset, referenced from
+stimulus_description if acquired data is present. Only present if
+not all parameter waves could be found.
+referenced:All referenced custom waves are stored here in a file-system like group-structure.
+/general/stimsets/referenced/ relates to root in the igor Experiment.
+epochs:
+tags:unused
+
Recent NWB (version 2) schema specifications are tracked in a separate
+repository. The
+schema is implemented in version 2.2.0
+(62c73400565afc28f67ede4f2e86023c33167cf8).
+
The complete schema tree is described in a hdmf compatible format and
+replicated in this repository under namespace/{schema}/{yaml,json}/*. The
+JSON files are stored in the nwb file upon storage. A build script exists to
+generate the JSON files from their YAML files: update_specifications.sh.
991ab801aae626710a8ec710869bb1b4508e91b7 ndx-MIES (heads/first-steps)
+24fba6174ddbad171ee5bb824edfa31f86b1b16d specifications (2.2.4)
+
+# diff namespace core upstream vs IPNWB specifications for nwb.base
+# diff namespace core upstream vs IPNWB specifications for nwb.behavior
+# diff namespace core upstream vs IPNWB specifications for nwb.device
+# diff namespace core upstream vs IPNWB specifications for nwb.ecephys
+# diff namespace core upstream vs IPNWB specifications for nwb.epoch
+# diff namespace core upstream vs IPNWB specifications for nwb.file
+# diff namespace core upstream vs IPNWB specifications for nwb.icephys
+# diff namespace core upstream vs IPNWB specifications for nwb.image
+# diff namespace core upstream vs IPNWB specifications for nwb.misc
+# diff namespace core upstream vs IPNWB specifications for nwb.namespace
+2,4c2
+< - name: core
+< doc: NWB namespace
+< author:
+---
+> - author:
+17a16
+> doc: NWB namespace
+18a18
+> name: core
+23c23
+< source: nwb.base.yaml
+---
+> source: nwb.base
+26c26
+< source: nwb.device.yaml
+---
+> source: nwb.device
+29c29
+< source: nwb.epoch.yaml
+---
+> source: nwb.epoch
+32c32
+< source: nwb.image.yaml
+---
+> source: nwb.image
+35c35
+< source: nwb.file.yaml
+---
+> source: nwb.file
+38c38
+< source: nwb.misc.yaml
+---
+> source: nwb.misc
+41c41
+< source: nwb.behavior.yaml
+---
+> source: nwb.behavior
+45c45
+< source: nwb.ecephys.yaml
+---
+> source: nwb.ecephys
+49c49
+< source: nwb.icephys.yaml
+---
+> source: nwb.icephys
+52c52
+< source: nwb.ogen.yaml
+---
+> source: nwb.ogen
+55c55
+< source: nwb.ophys.yaml
+---
+> source: nwb.ophys
+58c58
+< source: nwb.retinotopy.yaml
+---
+> source: nwb.retinotopy
+# diff namespace core upstream vs IPNWB specifications for nwb.ogen
+# diff namespace core upstream vs IPNWB specifications for nwb.ophys
+# diff namespace core upstream vs IPNWB specifications for nwb.retinotopy
+# diff namespace hdmf-common upstream vs IPNWB specifications for namespace
+2,4c2
+< - name: hdmf-common
+< doc: Common data structures provided by HDMF
+< author:
+---
+> - author:
+13a12
+> doc: Common data structures provided by HDMF
+14a14
+> name: hdmf-common
+17c17
+< source: table.yaml
+---
+> source: table
+20c20
+< source: sparse.yaml
+---
+> source: sparse
+# diff namespace hdmf-common upstream vs IPNWB specifications for sparse
+# diff namespace hdmf-common upstream vs IPNWB specifications for table
+# diff namespace ndx-mies upstream vs IPNWB specifications for namespace
+# diff namespace ndx-mies upstream vs IPNWB specifications for ndx-mies.extensions
+
+
+
The most important core properties are for intracellular ephys.
+
groups:
+-neurodata_type_def:PatchClampSeries
+neurodata_type_inc:TimeSeries
+doc:An abstract base class for patch-clamp data - stimulus or response,
+current or voltage.
+attributes:
+-name:stimulus_description
+dtype:text
+doc:Protocol/stimulus name for this patch-clamp dataset.
+-name:sweep_number
+dtype:uint32
+doc:Sweep number, allows to group different PatchClampSeries together.
+required:false
+datasets:
+-name:data
+dtype:numeric
+dims:
+-num_times
+shape:
+-null
+doc:Recorded voltage or current.
+attributes:
+-name:unit
+dtype:text
+doc:Base unit of measurement for working with the data. Actual stored values are
+not necessarily stored in these units. To access the data in these units,
+multiply 'data' by 'conversion'.
+-name:gain
+dtype:float32
+doc:Gain of the recording, in units Volt/Amp (v-clamp) or Volt/Volt (c-clamp).
+quantity:'?'
+links:
+-name:electrode
+target_type:IntracellularElectrode
+doc:Link to IntracellularElectrode object that describes the electrode that was
+used to apply or record this data.
+
+-neurodata_type_def:CurrentClampSeries
+neurodata_type_inc:PatchClampSeries
+doc:Voltage data from an intracellular current-clamp recording. A
+corresponding CurrentClampStimulusSeries (stored separately as a stimulus) is
+used to store the current injected.
+datasets:
+-name:data
+doc:Recorded voltage.
+attributes:
+-name:unit
+dtype:text
+value:volts
+doc:Base unit of measurement for working with the data. which is fixed to 'volts'.
+Actual stored values are not necessarily stored in these units. To access the data in these units,
+multiply 'data' by 'conversion'.
+-name:bias_current
+dtype:float32
+doc:Bias current, in amps.
+quantity:'?'
+-name:bridge_balance
+dtype:float32
+doc:Bridge balance, in ohms.
+quantity:'?'
+-name:capacitance_compensation
+dtype:float32
+doc:Capacitance compensation, in farads.
+quantity:'?'
+
+-neurodata_type_def:IZeroClampSeries
+neurodata_type_inc:CurrentClampSeries
+doc:Voltage data from an intracellular recording when all current
+and amplifier settings are off (i.e., CurrentClampSeries fields will be zero).
+There is no CurrentClampStimulusSeries associated with an IZero series because
+the amplifier is disconnected and no stimulus can reach the cell.
+datasets:
+-name:bias_current
+dtype:float32
+value:0.0
+doc:Bias current, in amps, fixed to 0.0.
+-name:bridge_balance
+dtype:float32
+value:0.0
+doc:Bridge balance, in ohms, fixed to 0.0.
+-name:capacitance_compensation
+dtype:float32
+value:0.0
+doc:Capacitance compensation, in farads, fixed to 0.0.
+
+-neurodata_type_def:CurrentClampStimulusSeries
+neurodata_type_inc:PatchClampSeries
+doc:Stimulus current applied during current clamp recording.
+datasets:
+-name:data
+doc:Stimulus current applied.
+attributes:
+-name:unit
+dtype:text
+value:amperes
+doc:Base unit of measurement for working with the data. which is fixed to 'amperes'.
+Actual stored values are not necessarily stored in these units. To access the data in these units,
+multiply 'data' by 'conversion'.
+
+-neurodata_type_def:VoltageClampSeries
+neurodata_type_inc:PatchClampSeries
+doc:Current data from an intracellular voltage-clamp recording. A
+corresponding VoltageClampStimulusSeries (stored separately as a stimulus) is
+used to store the voltage injected.
+datasets:
+-name:data
+doc:Recorded current.
+attributes:
+-name:unit
+dtype:text
+value:amperes
+doc:Base unit of measurement for working with the data. which is fixed to 'amperes'.
+Actual stored values are not necessarily stored in these units. To access the data in these units,
+multiply 'data' by 'conversion'.
+-name:capacitance_fast
+dtype:float32
+doc:Fast capacitance, in farads.
+quantity:'?'
+attributes:
+-name:unit
+dtype:text
+value:farads
+doc:Unit of measurement for capacitance_fast, which is fixed to 'farads'.
+-name:capacitance_slow
+dtype:float32
+doc:Slow capacitance, in farads.
+quantity:'?'
+attributes:
+-name:unit
+dtype:text
+value:farads
+doc:Unit of measurement for capacitance_fast, which is fixed to 'farads'.
+-name:resistance_comp_bandwidth
+dtype:float32
+doc:Resistance compensation bandwidth, in hertz.
+quantity:'?'
+attributes:
+-name:unit
+dtype:text
+value:hertz
+doc:Unit of measurement for resistance_comp_bandwidth, which is fixed to 'hertz'.
+-name:resistance_comp_correction
+dtype:float32
+doc:Resistance compensation correction, in percent.
+quantity:'?'
+attributes:
+-name:unit
+dtype:text
+value:percent
+doc:Unit of measurement for resistance_comp_correction, which is fixed to 'percent'.
+-name:resistance_comp_prediction
+dtype:float32
+doc:Resistance compensation prediction, in percent.
+quantity:'?'
+attributes:
+-name:unit
+dtype:text
+value:percent
+doc:Unit of measurement for resistance_comp_prediction, which is fixed to 'percent'.
+-name:whole_cell_capacitance_comp
+dtype:float32
+doc:Whole cell capacitance compensation, in farads.
+quantity:'?'
+attributes:
+-name:unit
+dtype:text
+value:farads
+doc:Unit of measurement for whole_cell_capacitance_comp, which is fixed to 'farads'.
+-name:whole_cell_series_resistance_comp
+dtype:float32
+doc:Whole cell series resistance compensation, in ohms.
+quantity:'?'
+attributes:
+-name:unit
+dtype:text
+value:ohms
+doc:Unit of measurement for whole_cell_series_resistance_comp, which is fixed to 'ohms'.
+
+-neurodata_type_def:VoltageClampStimulusSeries
+neurodata_type_inc:PatchClampSeries
+doc:Stimulus voltage applied during a voltage clamp recording.
+datasets:
+-name:data
+doc:Stimulus voltage applied.
+attributes:
+-name:unit
+dtype:text
+value:volts
+doc:Base unit of measurement for working with the data. which is fixed to 'volts'.
+Actual stored values are not necessarily stored in these units. To access the data in these units,
+multiply 'data' by 'conversion'.
+
+-neurodata_type_def:IntracellularElectrode
+neurodata_type_inc:NWBContainer
+doc:An intracellular electrode and its metadata.
+datasets:
+-name:description
+dtype:text
+doc:Description of electrode (e.g., whole-cell, sharp, etc.).
+-name:filtering
+dtype:text
+doc:Electrode specific filtering.
+quantity:'?'
+-name:initial_access_resistance
+dtype:text
+doc:Initial access resistance.
+quantity:'?'
+-name:location
+dtype:text
+doc:Location of the electrode. Specify the area, layer, comments on estimation
+of area/layer, stereotaxic coordinates if in vivo, etc. Use standard atlas
+names for anatomical regions when possible.
+quantity:'?'
+-name:resistance
+dtype:text
+doc:Electrode resistance, in ohms.
+quantity:'?'
+-name:seal
+dtype:text
+doc:Information about seal used for recording.
+quantity:'?'
+-name:slice
+dtype:text
+doc:Information about slice used for recording.
+quantity:'?'
+links:
+-name:device
+target_type:Device
+doc:Device that was used to record from this electrode.
+
+-neurodata_type_def:SweepTable
+neurodata_type_inc:DynamicTable
+doc:The table which groups different PatchClampSeries together.
+datasets:
+-name:sweep_number
+neurodata_type_inc:VectorData
+dtype:uint32
+doc:Sweep number of the PatchClampSeries in that row.
+-name:series
+neurodata_type_inc:VectorData
+dtype:
+target_type:PatchClampSeries
+reftype:object
+doc:The PatchClampSeries with the sweep number in that row.
+-name:series_index
+neurodata_type_inc:VectorIndex
+doc:Index for series.
+
+
+
The includes a Dynamic Table at /general/intracellular_ephys/sweep_table to
+store the sweep numbers of a list of data sets. The table is column centric
+and consists of the two columns sweep_number and series. Series contains
+links to datasets. The sweep_number for a dataset is stored under the same
+row index, specified either by the Dataset id (zero-based indices) or by
+series_index (one-based indices). The sweep_table is intended to easily find
+datasets that belong to a given sweep number. A sweep table is loaded by
+LoadSweepTable and created using AppendToSweepTable
Igor Pro module for reading and writing NeurodataWithoutBorder files¶
+
This modules allows to easily write and read valid NeurodataWithoutBorder style HDF5 files. It encapsulates the most commonly used
+parts of the specification in easy to use functions.
#pragmaTextEncoding="UTF-8"
+
+#defineIPNWB_DEFINE_IM
+#include"IPNWB_include"
+
+FunctionNWBWriterExample()
+
+variablefileID
+stringcontents,device
+variableversion= 1
+
+// Open a dialog for selecting an HDF5 file name
+HDF5CreateFilefileIDas""
+
+// fill gi/ti/si with appropriate data for your lab and experiment
+// if you don't care about that info just pass the initialized structures
+STRUCTIPNWB#GeneralInfogi
+STRUCTIPNWB#ToplevelInfoti
+STRUCTIPNWB#SubjectInfosi
+
+// takes care of initializing
+IPNWB#InitToplevelInfo(ti,version)
+IPNWB#InitGeneralInfo(gi)
+IPNWB#InitSubjectInfo(si)
+
+IPNWB#CreateCommonGroups(fileID,toplevelInfo=ti,generalInfo=gi,subjectInfo=si)
+
+// If you open an existing NWB file to append to, use the following command
+// to add an modification time entry, is implicitly called in IPNWB#CreateCommonGroups
+// IPNWB#AddModificationTimeEntry(locationID)
+
+// 1D waves from your measurement program
+// we use fake data here
+Make/FREE/N=1000AD=(sin(p)+cos(p/10))*enoise(0.1)
+SetScale/Px,0,5e-6,"s"
+
+// write AD data to the file
+STRUCTIPNWB#WriteChannelParamsparams
+IPNWB#InitWriteChannelParams(params)
+
+params.device="My Hardware"
+params.clampMode=0// 0 for V_CLAMP_MODE, 1 for I_CLAMP_MODE
+params.channelSuffix=""
+params.sweep= 123
+params.electrodeNumber= 1
+params.electrodeName="Nose of the mouse"
+params.stimset="My fancy sine curve"
+params.channelType=0// @see IPNWBChannelTypes
+WAVEparams.data=AD
+
+device="My selfbuilt DAC"
+
+IPNWB#CreateIntraCellularEphys(fileID)
+sprintfcontents,"Electrode %d",params.ElectrodeNumber
+IPNWB#AddElectrode(fileID,params.electrodeName,version,contents,device)
+
+// calculate the timepoint of the first wave point relative to the session_start_time
+// last time the wave was modified (UTC)
+params.startingTime=NumberByKeY("MODTIME",WaveInfo(AD,0))-date2secs(-1, -1, -1)
+params.startingTime-=ti.session_start_time// relative to the start of the session
+// we want the timestamp of the beginning of the measurement
+params.startingTime-=IndexToScale(AD,DimSize(AD, 0) - 1, 0)
+
+IPNWB#AddDevice(fileID,"Device name",version,"My hardware specs")
+
+STRUCTIPNWB#TimeSeriesPropertiestsp
+IPNWB#InitTimeSeriesProperties(tsp,params.channelType,params.clampMode)
+
+// all values not added are written into the missing_fields dataset
+IPNWB#AddProperty(tsp,"capacitance_fast", 1.0)
+IPNWB#AddProperty(tsp,"capacitance_slow", 1.0)
+
+// setting chunkedLayout to zero makes writing faster but increases the final filesize
+IPNWB#WriteSingleChannel(fileID,"/acquisition/timeseries",version,params,tsp)
+
+// write DA, stimulus presentation and stimulus template accordingly
+// ...
+
+// close file
+IPNWB#H5_CloseFile(fileID)
+End
+
Datasets which originate from Igor Pro waves have the special attributes
+IGORWaveScaling, IGORWaveType, IGORWaveUnits, IGORWaveNote. These attributes
+allow easy and convenient loading of the data into Igor Pro back.
+
For AD/DA/TTL groups the naming scheme is data_XXXXX_[AD/DA/TTL]
+suffix where XXXXX is a running number and suffix the channel
+number. For some hardware types the suffix includes the TTL line as
+well. It is important to note that the number of digits in XXXXX is
+variable und subject to change, and that XXXXX is not the sweep number.
+
In NWB v1, the sweep number is accessible from the source attribute only.
+Example source contents:
+Device=ITC18USB_Dev_0;Sweep=0;AD=0;ElectrodeNumber=0;ElectrodeName=0
+
For I=0 clamp mode neither the DA data nor the stimset is saved.
+
Some entries in the following tree are specific to MIES, these are marked as
+custom entries. Users running MIES are encouraged to use the same NWB layout
+and extensions.
The Sweep Formula Module in MIES_Sweepformula.ipf is intended to be used from
+the SF tab in the BrowserSettingsPanel (BSP). It is useful for analyzing a
+range of sweeps using pre-defined functions. The backend parses a formula into
+a JSON logic like pattern which in turn is analyzed
+to return a wave for plotting.
The entered code in the notebook is preprocessed. The preprocessor
+removes comments before testing the code for the ` vs ` operator after which
+it is passed to the formula parser.
+Comments start with a # character and end at the end of the current line.
In order for a formula to get executed, it has to be analyzed. This assures
+that the correct order of calculations is used. The approach for solving this
+is using a token based state machine. We virtually insert one character at a
+time from left to right into the state machine. Usually, a character is
+collected into a buffer. At some special characters like a + sign, the state
+changes from collect to addition. If a state changes, a new evaluation group is
+created which is represented with a JSON object who’s (single) member is the
+operation. The member name is the operation and the value is an ordered array
+of the operands. To ensure that multiplication is executed before addition to
+get 1+2*3=7 and not 1+2*3=9 the states have a priority. Higher order states
+cause the operation order to switch. The old operation becomes part of the new
+operation. In this context, when the first array or function argument separator ,
+is parsed on a level, it is treated as higher order operations because it creates
+a new array.
+
{
+"+":[
+1,
+{
+"*":[
+2,
+3
+]
+}
+]
+}
+
+
+
Arrays start with a square bracket [ and end with a ]. Subsequent array elements are
+separated by a ,. In a series of arrays like [1, 2], [3, 4], [5, 6] the , after
+the ] is enforced by the parser. Arrays can be part of arrays. Since at its core very
+formula input is an array the series of arrays [1, 2], [3, 4], [5, 6] is implicitly
+a 2-dimensional array: [[1, 2], [3, 4], [5, 6]]. The same applies for simple inputs like
+1, which is implicitly treated as 1-dimensional array: [1]. The input [[1]] instead
+is treated as 1x1 2-dimensional array.
+Arrays are special as
+also function arguments contain array elements. Therefore, an array can also
+simply be created by omitting the array brackets and only using element
+separators similar as in functions. The function max(1,2) is therefore
+treated the same as max([1,2]). Arrays can represent data and functions
+evaluate to arrays. Arrays can be of arbitrary size and can also be
+concatenated as in max(0,min(1,2),1).
A number can be entered as 1000, 1e3, or 10.0e2. It is always stored as a
+numeric value and not as string. The formula parser treats everything that is
+not parsable but matches alphanumeric characters (excluding operations) to a
+string as in a_string. White spaces are ignored throughout the
+formula which means that strings do not need to get enclosed by “. In fact,
+a “ is an disallowed character.
+
[
+1000,
+"a_string"
+]
+
+
+
A function is defined as a string that is directly followed by an opening
+parenthesis. The parenthesis token causes to force a collect state until all
+parentheses are closed.
+
Everything that is collected in a buffer is sent back to the function via
+recursive execution. The formula parser only handles elements inside one
+recursion call that are linearly combinable like 1*2+3*4. If same operations
+follow each other, they are concatenated into the same array level as for
+1+2+3+4.
The formula is sent to a preparser that checks for the correct
+amount of brackets and converts multi-character operations to their multi-character
+UTF-8 representations like … to …. It should be noted that an
+operation consists of one UTF-8 character. Functions on the other hand can
+consist of an arbitrary length of alphanumeric characters. The corresponding
+function for the above operation is range().
The formula executor receives a JSON id. It can only evaluate a specific
+structure of a formula which means for usual cases that it should start with an
+object that contains one operation. Operations are evaluated via recursive
+calls to the formula executor at different paths. This ensures that the formula
+is evaluated from the last element to the first element. The formula in the
+above example 1*2+3*4 is therefore parsed to
called from + operation -> evaluate /+ array to array with two elements
+
evaluate /+/0 to * operation with an array argument with two elements 1, 2
+
called from * operation -> evaluate /+/0/* array to wave {1, 2}
+
* operation is applied to wave {1, 2}, returning wave {2}
+
insert wave {2} as first element of array from step 2
+
evaluate /+/1 to * operation with an array argument with two elements 3, 4
+
called from * operation -> evaluate /+/0/* array to wave {3, 4}
+
* operation is applied to wave {3, 4}, returning wave {12}
+
insert wave {12} as second element of array from step 2
+
+ operation is applied to wave {2, 12} returning wave {14}
+
+
At the time of an evaluation, the maximum depth of an array is
+four dimensions as Igor Pro supports only four dimensions. This implies that on
+recursive evaluation of multi dimensional arrays the sub arrays can be
+three dimensional at best.
The array evaluation supports numeric and text data. The interpretation of the JSON arrays as
+text data is preferred. This means that [“NaN”] returns a one element text wave {“NaN”},
+whereas [1, “NaN”] returns a two element numeric wave {1, NaN}. If one element can not be
+parsed as string then it is assumed that the array contains numeric data.
+The JSON null element is only allowed for the topmost array as the parser inserts it for
+operation with no argument like e.g. select(). For sub arrays null elements [null]
+are invalid and result in an error.
+
If the topmost array is empty [] an empty numeric wave with zero size is returned.
+When checked in operation code the wave size should be checked before the wave type.
+
If the current array evaluated is of size one, then
+the wave note is transferred from the subArray to the current array. This is important for the case where the element of
+the current array is an JSON object, thus an operation, and the operation result is a single value with meta data in the wave.
The data is stored internally in persistant wave reference waves in a data folder, e.g.
+root:MIES:HardwareDevices:Dev1:Databrowser:FormulaData:. The reason is that operation like data(…)
+should be able to return multiple independent sweep data waves. These can be returned through a
+wave reference wave. Each wave referenced contains numeric or text data.
+The formula executor works on the JSON data that was created by the formula parser only.
+This data is by definition either an object (operation), numeric or a textual.
+If an operation like data(…) returns sweep data of multiple sweeps in a persistent wave reference wave
+for the formula executor a single element text wave is created.
+This text wave encodes a marker and the path to the wave reference wave in the first element.
+The wave reference wave is resolved by wrapper functions when calling the formula executor,
+such that the formula executor works only with the data wave(s).
+
Wrapper functions are:
+
+
SF_GetArgument: retrieves an operation argument, returns a wave reference wave. If in the JSON from the parser the argument consists of ‘direct’ data like an array then it is automatically converted to a wave reference wave with one element that refers to the data wave.
+
SF_GetArgumentSingle: retrieves an operation argument expecting only a single data wave. Returns the data wave.
+
SF_GetArgumentTop: retrieves all operation arguments as an array, returns a wave reference wave.
+
SF_GetOutputForExecutor: Takes a wave reference wave as input and creates a single element text wave for returning further to the formula executor.
+
SF_GetOutputForExecutorSingle: Takes a data wave as input, creates a single element wave reference wave referring to the data wave and creates text wave for returning further to the formula executor.
+
+
The wrapper function imply that the formula executor is never called directly from operation code.
+Also directly parsing the JSON is not allowed in operation code because every argument could be another operation or multi dimensional array etc.
By default only the currently used wave reference waves are persistent. For debugging the execution the SWEEPFORMULA_DEBUG define can be set:
+#define SWEEPFORMULA_DEBUG.
+When set all data waves and wave reference waves are stored persistently in the sweepformula working data folder that are created during the execution.
+The naming scheme is as follows: “source_pointOfCreation” with
+
+
source
typically the name of the operation or “ExecutorSubArrayEvaluation”
+
+
+
pointOfCreation:
+
+
output
wave reference wave output of operation
+
+
dataInput
data wave of direct data from JSON
+
+
refFromuserInput
wave reference wave automatically created to for data wave of direct data from JSON
+
+
return_argX_
data wave(s) returned by an operation, X counts the data waves aka index in the associated wave reference wave
+
+
argTop
prefix for the upper tags, added when data was parsed from the top level, used e.g. by integrate(1, 2)
+
+
+
The final wave name might be suffixed by a number guaranteeing unique wave names when multiple times the same operation was called.
In the context of the formula executor, different operations and functions are
+defined. Some of them are MIES specific, some of them are wrappers to Igor
+Pro operations or functions, some borrowed from other languages and there are
+also the simple, trivial operations. This section should give a list of the
+available operations and give a look into how they are meant to be used
+
The trivial operations are +, -, *, /. They are defined for all
+available dimensions and evaluate column based.
+
They can be used for evaluating
+
+
scalars with 1d waves as in 1 + [1, 2] = [1, 1] + [1, 2] = [2, 3]
+
1d waves with 1d waves as in [1, 2] + [3, 4] = [4, 6]
+
1d waves with 2d waves as in [1, 2] + [[3, 4], [5, 6]] = [[1 + 3, 2 + 5], [NaN + 4, NaN + 6]] = [[4, 7], [NaN, NaN]]
+
2d waves with 2d waves as in [[1, 2], [3, 4]] + [[5, 6], [7, 8]] = [[6, 8], [10, 12]]
+
+
The size in each dimension is expanded to match the maximum array size. The maximum array size is determined by the required maximum dimensions of the elements in the topmost array.
+An array element can be a number, a string, an array or an operation. A number or string a scalar. An sub array or operaton result is scalar if it returns a single element.
+The expansion is filled with for numeric waves with NaN or for textual waves with “”.
+In the special case of a scalar element, the value is expanded to the full size and dimensions of the expanded arrays size.
+This means that in our first example, 1 is scalar and is internally expanded to an array of size 2 because the second operand determines the maximum size of 2:
+1 + [1, 2] == [1, 1] + [1, 2].
+On the other hand in the third example above the first arrays size is expanded but not its value as it is not a scalar.
+The array size expansion and scalar elements expansion is applied recursively for more dimensions.
+Note that operations in array elements may return multi dimensional sub arrays that lead to an overall array expansion that is greater as the formula input suggests.
min and max return the minimum and maximum of an array.
+The operation takes 1 to N arguments. The input data must be 1d or 2d, numeric and have at least one data point.
+The operations work column based, such that for each column e.g. the maximum of all row values is determined. An 2d input array of size MxN is returned as 1d array of the size N.
+When called with a single argument the operation accepts multiple data waves.
+For this case the operation is applied on each input data wave independently and returns the same number of data waves.
+The returned data type is SF_DATATYPE_MIN or SF_DATATYPE_MAX.
+If input data type is SF_DATATYPE_SWEEP from the data operation the sweep meta data is transferred to the returned data waves.
+The default suggested x-axis values for the formula plotter are sweep numbers.
avg and mean are synonyms for the same operation.
+They calculate the arithmetic average \(\frac{1}{n}\sum_i{x_i}\).
+
data: input data wave(s)
+
+
mode: optional parameter that defines in which direction the average is applied.
+
in default, applies the average over each input data wave. In this mode the operation returns the same number of waves as input waves were specified. Each output wave contains a single data point. If input data type is SF_DATATYPE_SWEEP from the data operation the sweep meta data is transferred to the returned data waves. The default suggested x-axis values for the formula plotter are sweep numbers.
+
over averages over all input data waves. In this mode the operation returns a single wave. NaN values in input waves are ignored in the average calculation. A trace generated from the returned wave will be shown as topmost trace in the default color for averaged data.
rms calculates the root mean square \(\sqrt{\frac{1}{n}\sum_i{x_i^2}}\) of a row if the wave is 1d. It calculates column based if the wave is 2d.
+The operation takes 1 to N arguments. The input data must be 1d or 2d, numeric and have at least one data point.
+The operations works column based, such that for each column e.g. the average of all row values is determined. An 2d input array of size MxN is returned as 1d array of the size N.
+When called with a single argument the operation accepts multiple data waves.
+For this case the operation is applied on each input data wave independently and returns the same number of data waves.
+The returned data type is SF_DATATYPE_RMS.
+If input data type is SF_DATATYPE_SWEEP from the data operation the sweep meta data is transferred to the returned data waves.
+The default suggested x-axis values for the formula plotter are sweep numbers.
variance calculates the variance of a row if the wave is 1d. It calculates column based if the wave is 2d.
+Note that compared to the Igor Pro function variance() the operation does not ignore NaN or Inf.
+The operation takes 1 to N arguments. The input data must be 1d or 2d, numeric and have at least one data point.
+The operations works column based, such that for each column e.g. the average of all row values is determined. An 2d input array of size MxN is returned as 1d array of the size N.
+When called with a single argument the operation accepts multiple data waves.
+For this case the operation is applied on each input data wave independently and returns the same number of data waves.
+The returned data type is SF_DATATYPE_VARIANCE.
+If input data type is SF_DATATYPE_SWEEP from the data operation the sweep meta data is transferred to the returned data waves.
+The default suggested x-axis values for the formula plotter are sweep numbers.
stdev calculates the variance of a row if the wave is 1d. It calculates column based if the wave is 2d.
+The operation does not ignore NaN or Inf.
+The operation takes 1 to N arguments. The input data must be 1d or 2d, numeric and have at least one data point.
+The operations works column based, such that for each column e.g. the average of all row values is determined. An 2d input array of size MxN is returned as 1d array of the size N.
+When called with a single argument the operation accepts multiple data waves.
+For this case the operation is applied on each input data wave independently and returns the same number of data waves.
+The returned data type is SF_DATATYPE_STDEV.
+If input data type is SF_DATATYPE_SWEEP from the data operation the sweep meta data is transferred to the returned data waves.
+The default suggested x-axis values for the formula plotter are sweep numbers.
Use area to calculate the area below a 1D array using trapezoidal integration.
+
area(arraydata[,variablezero])
+
+
+
The first argument is the data, the second argument specifies if the data is zeroed. Zeroing refers to an additional differentiation and integration of the data prior the
+area calculation. If the zero argument is set to 0 then zeroing is disabled. By default zeroing is enabled.
+If zeroing is enabled the input data must have at least 3 points. If zeroing is disabled the input data must have at least one point.
+The operation ignores NaN in the data.
+The operations works column based, such that for each column e.g. the area of all row values is determined. An 2d input array of size MxN is returned as 1d array of the size N.
+An 3d input array of size MxNxO is returned as 2d array of the size NxO.
+The operation accepts multiple data waves for the data argument.
+For this case the operation is applied on each input data wave independently and returns the same number of data waves.
+The returned data type is SF_DATATYPE_AREA.
+If input data type is SF_DATATYPE_SWEEP from the data operation the sweep meta data is transferred to the returned data waves.
+The default suggested x-axis values for the formula plotter are sweep numbers.
Use derivative to differentiate along rows for 1- and 2-dimensional data.
+
derivative(arraydata)
+
+
+
Central differences are used. The same amount of points as the input is returned.
+The input data must have at least one point.
+The operation ignores NaN in the data.
+The operation accepts multiple data waves for the data argument.
+For this case the operation is applied on each input data wave independently and returns the same number of data waves.
+The returned data type is SF_DATATYPE_DERIVATIVE.
Use integrate to apply trapezoidal integration along rows. The operation returns the same number of points as the input wave(s).
+
integrate(arraydata)
+
+
+
Note that due to the end point problem it is not the counter-part of derivative.
+The input data must have at least one point.
+The operation ignores NaN in the data.
+The operation accepts multiple data waves for the data argument.
+For this case the operation is applied on each input data wave independently and returns the same number of data waves.
+The returned data type is SF_DATATYPE_INTEGRATE.
The operation butterworth applies a butterworth filter on the given data
+using FilterIIR from Igor Pro. The operation calculates along rows. It takes
+four arguments:
The first parameter data is intended to be used with the data() operation but
+can be an arbitrary numeric array. The parameters lowPassCutoffInHz and
+highPassCutoffInHz must be given in Hz. The maximum value for order is 100.
+The operation accepts multiple data waves for the data argument.
+For this case the operation is applied on each input data wave independently and returns the same number of data waves.
+The returned data type is SF_DATATYPE_BUTTERWORTH.
The function xvalues or time are synonyms for the same function.
+The function returns a wave containing the x-scaling of the
+input data.
+
xvalues(arraydata)
+
+
+
The output data wave has the same dimension as the input data. The x-scaling values are filled in the rows for all dimensions.
+The operation accepts multiple data waves for the data argument.
+For this case the operation is applied on each input data wave independently and returns the same number of data waves.
dimension where the scale should be set, either d, x, y, z or t.
+
+
dimOffset
optional, the scale offset for the first data point. If not specified, 0 is used as default.
+
+
dimDelta
optional, the scale delta for the data point distance. If not specified, 1 is used as default.
+
+
unit
optional, the scale unit for the data points. If not specified, “” is used as default.
+
+
+
If d is used for dim, then in analogy to Igor Pros SetScale operation the dimOffset and dimDelta argument set the nominal minimum and nominal maximum data values of the wave.
+
If x, y, z or t is used for dim and dimDelta is 0 then the default dimDelta 1 is used.
+
The operation accepts multiple data waves for the data argument.
+For this case the operation is applied on each input data wave independently and returns the same number of data waves.
The operation selchannels allows to select channels.
+selchannels([str name]+) converts named channels from strings to numbers.
+
The function accepts an arbitrary amount of channel names like AD, DA or
+TTL with a combination of numbers AD1 or channel numbers alone like 2.
+The maximum allowed channel number is NUM_MAX_CHANNELS (16). For all channel
+types the channel numbers as given on the DAEphys panel are accepted.
+The operation returns a numeric array of [[channelType+], [channelNumber+]] that has as
+row dimension the number of the input strings.
+When called without argument all channel types / channel numbers are set by setting the
+returned value for type and number to NaN.
+The result of selchannels has a data type attributed.
+
selchannels is intended to be used with the select() operation.
The operation selsweeps allows to select sweeps by their number and returns an 1d-array with the sweep numbers.
+The operation accepts numbers, arrays and ranges as arguments. Any number of arguments can be specified.
+In case no argument is given, then all sweeps are returned or if there are no sweeps a null wave is returned.
+Each unique sweep number is returned only once.
+The result of selsweeps has a data type attributed.
+
selsweeps is intended to be used with the select() operation.
+
# For this example two sweeps were acquired
+selsweeps()==[0,1]
+
+selsweeps(0)==0
+
+selsweeps([1,0])==[1,0]
+
+selsweeps(0...2)==[0,1]
+
+# For this example 30 sweeps were acquired
+selsweeps(10,[20,24],26...30)==[10,20,24,26,27,28,29]
+
+# Each unique sweep number is returned only once
+selsweeps(0,0,1)==[0,1]
+
The operation selrange allows to specify a time interval either by epoch name or numbers in ms.
+It takes zero or one argument, an epoch name/wildcard or an array with two
+numeric values. The numeric values specify the start and end of a range.
+The operation returns a dataset with a range specification array.
+In case no argument is given, then a dataset with a full-range specification is returned.
+The result of selrange has a data type attributed.
+
selrange is intended to be used with the select() operation.
+
# returns a full-range
+selrange()
+
+# refers to epoch E1
+selrange(E1)
+
+# all stimset epochs
+selrange("E*")
+
+# refers to 30 ms to 100 ms
+selrange([30,100])
+
+# refers to the range set by cursor A and B
+selrange(cursors(A,B))
+
The operation selvis allows to specify if selected data is taken from all sweeps or from the displayed sweeps only.
+It takes zero or one argument that can be either all or displayed.
+In case no argument is given, then the operation defaults to displayed.
+The operation returns a text wave with a single element.
+The result of selvis has a data type attributed.
+
selvis is intended to be used with the select() operation.
+
# refers to displayed
+selvis()
+
+# refers to all
+selvis(all)
+
+# refers displayed
+selvis(displayed)
+
The operation selcm allows to specify how select filters data for clamp mode.
+It takes between zero and any number of arguments.
+Allowed arguments are none, ic, vc, izero, all. If no argument is given then selcm defaults to all.
+The operation returns a a numeric value with a clamp code that is a logical ORed result of the given clamp modes.
+The result of selcm has a data type attributed.
+
selcm is intended to be used with the select() operation.
+
# sweep data in any clamp mode
+selcm()
+
+# sweep data acquired in current clamp and voltage clamp mode
+selcm(ic,vc)
+
+# sweep data acquired with no clamp mode (unassociated channels)
+selcm(none)
+
The operation selstimset allows to specify how select filters data regarding the stimset wave name.
+It takes between zero and any number of arguments.
+Allowed arguments are strings that can contain wildcards.
+If no argument is given then selstimset defaults to *.
+The operation returns a a text wave with the stimset wave name wildcard patterns.
+The result of selstimset has a data type attributed.
+
selstimset is intended to be used with the select() operation.
+
# sweep data with any stimset wave name
+selstimset()
+
+# sweep data with all stimset wave names that start with pinky and all that end with brain
+selstimset("pinky*","*brain")
+
The operation selexp allows to specify how select filters data regarding the experiment name.
+It takes exactly one string argument. The string that can contain wildcards.
+The operation returns a a text wave with the experiment name wildcard pattern.
+The result of selexp has a data type attributed.
+
selexp is intended to be used with the select() operation.
+
# sweep data from a specific experiment
+selexp("MyFirstExperiment.pxp")
+
+# sweep data from a specific experiment
+selexp("MySecondExp*")
+
The operation seldev allows to specify how select filters data regarding the DAC device name.
+It takes exactly one string argument. The string that can contain wildcards.
+The operation returns a a text wave with the device name wildcard pattern.
+The result of seldev has a data type attributed.
+
seldev is intended to be used with the select() operation.
+
# sweep data from a specific device
+seldev("ITC18*")
+
+# sweep data from a specific device
+seldev("Dev*")
+
When the operation selsetcyclecount is used with select it includes all sweeps with the specified set cycle count.
+The operation takes exactly one numerical argument.
+The operation returns a a numeric wave with a single element that has the value of the given argument.
+The result of selsetcyclecount has a data type attributed.
+
selsetcyclecount is intended to be used with the select() operation.
+
# sweeps that have a set cycle count of 5
+selsetcyclecount(5)
+
When the operation selsetsweepcount is used with select it includes all selection with the specified set sweep count.
+The operation takes exactly one numerical argument.
+The operation returns a a numeric wave with a single element that has the value of the given argument.
+The result of selsetsweepcount has a data type attributed.
+
selsetsweepcount is intended to be used with the select() operation.
+
# sweeps that have a set sweep count of 2
+selsetsweepcount(2)
+
When the operation selsciindex is used with select it includes all selections that have the n-th unique stimset cycle id.
+The specific order of the stimset cycle ids before this operation is applied depends on the other select filters applied in the select operation.
+Selections with no stimset cycle id are discarded and not indexed.
+The stimset cycle id depends on the headstage and thus, on channel type and channel number of the specific sweep.
+The selection results are determined per headstage. Thus, if the other select filters result
+in selections include multiple headstages then the n-th unique stimset cycle id is selected for each headstage seperately.
+Selections are sorted by the following priority list (higher to lower): experiment name, sweep number, channel type, channel number.
+The operation takes exactly one numerical argument.
+The operation returns a a numeric wave with a single element that has the value of the given argument.
+The result of selsciindex has a data type attributed.
+
selsciindex is intended to be used with the select() operation.
+
# Looks at all sweep starting from sweep 3 with channel AD0. Selects all sweeps that have starting from sweep 3 the third unique stimset cycle id.
+select(selsweeps([3,1000]),selchannels(AD0),selsciindex(3))
+
+# example, where the first three columns are the result of a selection, the last two columns are added for illustration
+# a possible selection with a two headstage setup could be select(selvis(all), selsweeps([0, 3]), selchannels(AD))
+SweepChannelTypeChannelNumerHeadstageStimsetCycleId
+0AD6043
+0AD7145
+1AD6043
+1AD7146
+2AD6044
+2AD7146
+
+# if based on this selection selsciindex(0) is applied:
+# select(selvis(all), selsweeps([0, 3]), selchannels(AD), selsciindex(0))
+# The result is
+SweepChannelTypeChannelNumerHeadstageStimsetCycleId
+0AD6043
+1AD6043
+0AD7145
+# for headstage 0 the 0-th SCI index is 43
+# for headstage 1 the 0-th SCI index is 45
+
+# if based on this selection selsciindex(1) is applied:
+# select(selvis(all), selsweeps([0, 3]), selchannels(AD), selsciindex(1))
+# The result is
+SweepChannelTypeChannelNumerHeadstageStimsetCycleId
+2AD6044
+1AD7146
+2AD7146
+# for headstage 0 the 1-th SCI index is 44
+# for headstage 1 the 1-th SCI index is 46
+
When the operation selracindex is used with select it includes all selections that have the n-th unique repeated acquisition cycle id.
+The specific order of the repeated acquisition cycle ids before this operation is applied depends on the other select filters applied in the select operation.
+Selections with no repeated acquisition cycle ids are discarded and not indexed.
+The selections prior to the application of selracindex are sorted by the following priority list (higher to lower): experiment name, sweep number, channel type, channel number.
+The operation takes exactly one numerical argument.
+The operation returns a a numeric wave with a single element that has the value of the given argument.
+The result of selracindex has a data type attributed.
+
selracindex is intended to be used with the select() operation.
+
# Looks at all sweep starting from sweep 3 with channel AD0. Selects all sweeps that have starting from sweep 3 the third unique repeated acquisition cycle id.
+select(selsweeps([3,1000]),selchannels(AD0),selracindex(3))
+
When the operation selexpandsci is used with select then select operates in a two-step regime.
+First the common select filters e.g. by sweep number, stimset, etc. are applied.
+Then for each of these selections the selections with the same stimset cycle id are also added.
+For example when a single sweep/channel is selected all other sweeps from the same stimset cycle id can be collected for the resulting selections.
+Intersections with additional selections from another select are applied afterwards.
+The operation takes no argument.
+The result of selexpandsci has a data type attributed.
+
selexpandsci is intended to be used with the select() operation.
+
# Looks at all AD channels from sweep 1 and selects all sweeps with the same stimset cycle id.
+select(selsweeps(1),selchannels(AD),selexpandsci())
+
When the operation selexpandrac is used with select then select operates in a two-step regime.
+First the common select filters e.g. by sweep number, stimset, etc. are applied.
+Then for each of these selections the selections with the same repeated acquisition cycle are also added.
+So for example when a single sweep/channel is selected all other sweeps from the same repeated acquisition cycle can be collected for the resulting selections.
+Intersections with additional selections from another select are applied afterwards.
+The operation takes no argument.
+The result of selexpandrac has a data type attributed.
+
selexpandrac is intended to be used with the select() operation.
+
# Looks at all AD channels from sweep 1 and selects all sweeps from the same repeated acquisition cycle.
+select(selsweeps(1),selchannels(AD),selexpandrac())
+
The operation selivsccsweepqc allows to specify how select filters data
+regarding the sweep quality check from the IVSCC analysis functions.
+It takes between one argument that can be either failed or passed.
+The operation returns a a text wave with the argument value as string.
+The result of selivsccsweepqc has a data type attributed.
+
selivsccsweepqc is intended to be used with the select() operation.
+
# sweep data where the analysis function passed the sweepqc check
+selivsccsweepqc(passed)
+
+# sweep data where the analysis function failed the sweepqc check
+selivsccsweepqc(failed)
+
The operation selivsccsetqc allows to specify how select filters data
+regarding the set quality check from the IVSCC analysis functions.
+It takes between one argument that can be either failed or passed.
+The operation returns a a text wave with the argument value as string.
+The result of selivsccsetqc has a data type attributed.
+
selivsccsetqc is intended to be used with the select() operation.
+
# sweep data where the analysis function passed the setqc check
+selivsccsetqc(passed)
+
+# sweep data where the analysis function failed the setqc check
+selivsccsetqc(failed)
+
The cursors operation returns the x-values of the named cursor(s).
+
cursors([A-J]+)
+
+
+
The cursors operation takes any number of arguments. If no argument is given
+it defaults to cursors(A, B). When cursors is used as argument for a range specification, e.g. for data
+two arguments for cursors should be used to have a compatible output.
+Valid cursor names are A-J. The operation returns a numeric 1d-wave containing the x-values of the named cursor(s).
+If a named cursor is not present, then NaN is returned as position.
The wave operation returns the content of the referenced wave.
+
wave(stringpathToWave)
+
+
+
If no wave can be resolved at the given path a null wave is returned. The further handling depends how the operations receiving such null wave handles this special case.
+The formula plotter skips null waves.
The operation text converts the given numeric data to a text data.
+
text(arraydata)
+
+
+
This can be used to force, for example, a category plot.
+text requires numeric input data. The output data has the same dimension as the input data. The output precision for the text are 7 digits after the dot.
+The operation accepts multiple data waves for the data argument.
+For this case the operation is applied on each input data wave independently and returns the same number of data waves.
It takes one argument that is either a select operation or an array of select operations.
+
If an array of select operations is specified then over each selected data is iterated
+independently. Thus, one data expression can retrieve sweep data from
+multiple select operations.
+
A given selrange in select as numbers or epoch extracts a subrange of data points from the sweep. The start and end time is converted to
+closest integer indices, where the included points range from startIndex to endIndex - 1. This matches the general handling
+of epochs in MIES, where the data point at the end time of an epoch is not part of the epoch range.
+
For each selected sweep/channel combination data returns a data wave. The data wave contains the sweep data for the specified range/epoch.
+If no sweep/channel was selected then the number of returned data waves is zero. Each data wave gets meta data about the originating sweep/channel added.
+The returned data type is SF_DATATYPE_SWEEP.
+
# AD channels of all displayed sweeps with the range 0 - 1s
+data(select(selrange([0,1000]),selchannels(AD)))
+
+# epoch "E1" range of the AD channels of all displayed sweeps
+data(select(selrange(E1),selchannels(AD)))
+
+# epoch "E1" range with the start offsetted by 10ms of the AD channels of all displayed sweeps
+sel=select(selchannels(AD))
+rng=epochs("E1",$sel)+[10,0]
+data(select(selrange($rng),$sel))
+
+# sweep data from all epochs starting with "E" of the AD channels of all displayed sweeps
+data(select(selrange("E*"),selchannels(AD)))
+
+# sweep data from all epochs starting with "E" and "TP" of the AD channels of all displayed sweeps
+sel1=select(selchannels(AD))
+sel2=select(selrange("E*"),$sel1)
+sel3=select(selrange("TP*"),$sel1)
+data([$sel2,$sel3])
+
+# sweep data from all epochs that do not start with "E" and that do start with "TP" of the AD channels of all displayed sweeps
+sel1=select(selchannels(AD))
+sel2=select(selrange("!E*"),$sel1)
+sel3=select(selrange("TP*"),$sel1)
+data([$sel2,$sel3])
+
+# extract the first pulse from TTL1 as epoch and extract the AD data in that range
+sel1=select(selchannels(TTL1))
+ep=epochs(E0_PT_P0,$sel1)
+data(select(selrange($ep),selchannels(AD)))
+
+# extract the first pulse from TTL1 as epoch with a start and end offset, then extract the AD data in that range
+sel1=select(selchannels(TTL1))
+ep=epochs(E0_PT_P0,$sel1)+[50,100]
+data(select(selrange($ep),selchannels(AD)))
+
+# filter by channel, clamp mode and stimset wave name, then based on that selection create one with epoch E0 and another with epoch E1 range
+# retrieve data for these two selections
+sel1=select(selchannels(AD),selcm(ic),selstimsets("AD_phase0*"))
+sel2=select(selrange(E0),$sel1)
+sel3=select(selrange(E1),$sel1)
+data([$sel2,$sel3]])
+
The labnotebook function returns the (case insensitive) key entry from the
+labnotebook for the selected channel and sweep combination(s). For selectData
+either a single select or an array of select’s can be specified. If an array
+is specified then over each selection is iterated independently.
+The optional entrySourceType can be one of the constants DataAcqModes for data
+acquisition modes as defined in ../MIES/MIES_Constants.ipf. If the
+entrySourceType is omitted it defaults to DATA_ACQUISITION_MODE. Wildcard
+expressions using */! are also supported. See here
+for a list of stock labnotebook entries.
+
When the optional select argument is omitted, select() is used as default
+that includes all displayed sweeps and channels.
+
The labnotebook operation returns a data wave for each selected sweep/channel
+combination. Each data wave contains a single element, that is depending on the
+requested labnotebook entry numerical or textual.
+
The returned data type is SF_DATATYPE_LABNOTEBOOK. If input data type is
+SF_DATATYPE_SWEEP from the data operation the sweep meta data is transferred
+to the returned data waves. The default suggested x-axis values for the formula
+plotter are sweep numbers.
The anafuncparam function returns the values of the requested analysis function
+parameters for the selected channel and sweep combination(s). For selectData
+either a single select or an array of select’s can be specified. If an array
+is specified then over each selection is iterated independently.
+Wildcard expressions using */! are also supported.
+See here for a list of parameters
+from analysis functions shipped with MIES.
+
When the optional select argument is omitted, select() is used as default
+that includes all displayed sweeps and channels.
+
The returned data type is SF_DATATYPE_ANAFUNCPARAM. The default suggested
+x-axis values for the formula plotter are sweep numbers.
one or multiple data waves. If multiple data waves are given then the same number of data waves is returned. The operation is applied for each data wave separately.
+
+
level
level value to find
+
+
edge
defines which transition is to be found. Valid values are rising and falling 0, rising 1 or falling 2. The default for edge is rising and falling 0.
+
+
+
The returned data type is SF_DATATYPE_FINDLEVEL.
+If input data type is SF_DATATYPE_SWEEP from the data operation the sweep meta data is transferred to the returned data waves.
one or multiple data waves. If multiple data waves are given then the same number of data waves is returned. The operation is applied for each data wave separately.
+
+
method
the method can be either
+
+
0 for “full”
+
1 for “instantaneous”
+
2 for apcount
+
3 for “instantaneous pair”
+
+
The default method is 0.
+
+
level
level threshold for peak detection. The level refers to the amplitude of the sweep(s). level is a numeric value and defaults to 0.
+
+
resultType
the result type defines what result(s) the apfrequency operation returns if the method 3 (instantaneous pair) is set.
+
+
time returns time intervals
+
freq returns frequencies.
+
+
+
normalize
sets the way the results get normalized
+
+
nonorm: no normalzation is applied (default)
+
normoversweepsmin: normalizes over all sweeps based on the minimum result value in all sweeps based on the current method
+
normoversweepsmax: normalizes over all sweeps based on the maximum result value in all sweeps based on he current method
+
normoversweepsavg: normalizes over all sweeps based on the average result value in all sweeps based on the current method
+
norminsweepsmin: normalizes each sweep based on the minimum result value in the specific sweep based on the current method
+
norminsweepsmax: normalizes each sweep based on the maximum result value in the specific sweep based on the current method
+
norminsweepsavg: normalizes each sweep based on the average result value in the specific sweep based on the current method
+
+
+
xAxisType
if the method 3 (instantaneous pair) is set then xAxisType defines the x-axis of the data display.
+
+
time: the x-axis shows the occurence in time of the first peak of the pair(s), default
+
count: the x-axis counts the pair(s)
+
+
+
+
The basic calculation for these methods are done using the below formulas where
+\(l\) denotes the number of found levels, \(t_{i}\) the timepoint in
+seconds of the level and \(T\) the total x range of the data in seconds.
The method 2 (instantaneous) and 3 (instantaneous pair) treat the peaks as interleaved pairs of peaks and returns results only if there are two or more peaks found.
+
The returned data type is SF_DATATYPE_APFREQUENCY. If input data type is
+SF_DATATYPE_SWEEP from the data operation the sweep meta data is transferred
+to the returned data waves. There is no input data verification, so it is left
+to the user to select a reasonable range or epoch.
the unit can be either default, dB for decibel or normalized for the spectrum normalized by its total energy. The default method is default.
+default means e.g. if the signal unit is V then the y-axis unit of the power spectrum is V^2.
+
+
average
this argument allows to enable averaging over all sweeps of the same channel/channeltype combination. Possible values are avg and noavg.
+The default average setting is noavg. If data waves do not originate from a sweep, then it is averaged over all of these data waves.
+e.g. if there are two data waves from sweep 0,1 AD1, two data waves from sweep 0,1 AD2 and two data waves not from a sweep then
+there will be three averaged waves: over all sweeps for channel combination AD1, over all sweeps for channel combination AD2 and over all data waves not from a sweep.
+
+
ratioFrequency
this argument allows to specify a frequency where the ratio between base line and signal is determined through a gaussian fit with a linear base.
+A typical use is to look for line noise at 50 Hz or 60 Hz. If a non zero ratioFrequency is set then the result is a single data point per power spectrum wave.
+The returned ratio is (amplitude + baseline_level) / baseline_level. The default ratioFrequency is 0, that disables the ratio determination.
+
+
cutOffFrequency
The cutOffFrequency allows to limit the maximum displayed frequency of the powerspectrum. The default cutOffFrequncy is 1000 Hz.
+The maximum cutOffFrequency is half of the sample frequency. This argument is ignored if a ratioFrequency > 0 is set.
+
+
windowFunction
allows to specify the window function applied for the FFT. The default windowFunction is Hanning.
+Possible options are none to disable the application of a window function and the window functions known to Igor Pro 9. See DisplayHelpTopic “FFT”.
+
+
+
The gaussian fit for the power ratio calculation uses the following constraints:
+
+
The peak position must be between ratioFrequency ± 0.25 Hz
+
The maximum FWHM are 5 Hz
+
The amplitude must be >= 0
+
The base of the peak must be > 0
+
+
If the fit fails a ratio of 0 is returned.
+
The returned data type is SF_DATATYPE_POWERSPECTRUM.
+If input data type is SF_DATATYPE_SWEEP from the data operation and non-averaged power spectrum is calculated the sweep meta data is transferred to the returned data waves.
The psxDeconvFilter operation is a helper operation for psx to manage the deconvolution filter settings.
+
+
psxDeconvFilter([lowFreq, highFreq, order])
+
+
The function accepts zero to three arguments.
+
+
lowFreq [Hz]
defaults to NaN
+
+
highFreq [Hz]
defaults to NaN
+
+
order
defaults to NaN
+
+
+
The default values of NaN are replaced inside psx. For the order this is
+101, for the frequencies this is a normalized frequency which depends on the
+sampling interval of the data. Here lowFreq is the end of the passband and
+highFreq the start of the reject band see also the description of /LO from
+FilterFIR.
Plot properties of the result waves of a miniature PSC/PSP classification. The
+operation combines the data from all input sweeps.
+
The operation allows to visualize psx data from the results wave or locally,
+i.e. from an psx operation from another formula separated by and. The
+local results are prefered over the results wave.
+
The traces are colored using the common headstage colors. The markers are the
+same as used for visualizing the event state in psx (accepted -> circle,
+rejected -> triangle, undetermined -> square).
identifier string, must adhere to strict igor object names.
+Used for identifying the data to query, also from the results wave
+
+
select
selections and range to operate on from the select operation
+
+
prop
column of the psx event results waves to plot.
+Choices are: amp, xpos, xinterval, tau, estate, fstate, fitresult, risetime
+
+
state
QC state to select the events.
+Choices are: accept/reject/undetermined/all/every
+
The used QC state depends on prop:
+
+
Event state QC -> amp/xpos/xinterval/estate/risetime
+
Fit state QC -> tau/fstate/fitresult
+
+
The difference between all and every is that all plots the events from
+all possible states in one trace whereas every creates multiple
+traces, one for each state.
+
+
postproc
post process the results, defaults to nothing
+Choices are: nothing, stats, nonfinite, count, hist, log10
+
+
nothing
no post processing
+
+
stats
calculate various statistical properties of the data
+
+
nonfinite
selects non-finite values (-inf/NaN/inf)
+
+
count
count the number of data elements
+
+
hist
create a histogram from the data
+
+
log10
apply the decadic logarithm (base 10) to each data point
The fit operation allows to perform a CurveFit on the given x and y data and
+accepts exactly three parameters.
+
fit(arraysxdata,arraysydata,fitOp)
+
+
+
+
xdata, ydata
one or multiple arrays with data
+
+
fitOp
helper operation with fit type and possible constrained parameters,
+currently only fitline is available.
+
+
+
xdata and ydata all need to be 1D, but multiple can be given. The
+number of points in the corresponding x and y waves must be the same.
+
Example:
+
# we look at four sweeps
+sweeps=[5,7,8,10]
+
+# grab the DA data from channel 0 and epoch E1
+selDA=select(selchannels(DA0),selsweeps($sweeps))
+dDA=data("E1",$selDA)
+
+# E2 from AD channel 2
+selAD=select(selchannels(AD2),selsweeps($sweeps))
+dAD=data("E2",$selAD)
+
+# calculate minimum for the data in each sweep,
+# but merge the data into one wave for the fit
+setX=merge(min($dDA))
+
+# and average for AD
+setY=merge(avg($dAD))
+
+# plot the extracted data
+$dDA
+
+and
+
+$dAD
+
+and
+
+# and the input data
+$setYvs$setX
+
+with
+
+# and do the fit
+fit($setX,$setY,fitline())
+
The select operation allows to choose a selection of sweep data from given filter operations.
+It is intended to be used with operations like data, labnotebook, epochs, tp and select itself.
+
select(filter,filter,...)
+
+
+
The function accepts any number of arguments from filter operations.
Sweeps that fit all filter criteria are taken into the selection. Each filter operation except select may appear once as argument.
+It is not required that the arguments have a specific order.
+
If a specific filter is not part of the arguments and none of the arguments is a select then default values are used:
+- selchannels: select all channels
+- selsweeps: select all sweep numbers
+- selrange: select full range
+- selvis: select displayed sweeps
+- selscm: select all clamp modes
+- selstimset: select all stimset wave names
+- selivsccsetqc: IVSCC SetQC is ignored
+- selivsccsweepqc: IVSCC SweepQC is ignored
+- selexp: experiment name is ignored
+- seldev: device name is ignored
+- selsetcyclecount: set cycle count is ignored
+- selsetsweepcount: set sweep count is ignored
+- selsciindex: stimset cycle id index is ignored
+- selracindex: repeated acquisition is index is ignored
+- selexpandrac: expansion by repeated acquisition cycle is disabled
+- selexpandsci: expansion by stimset cycle id is disabled
+
If a specific filter is not part of the arguments and there exists at least one arguments that is a select then these filters will be ignored:
+- selchannels: select all channels
+- selsweeps: select all sweep numbers
+- selrange: select full range
+- selvis: select all sweeps
+- selscm: select all clamp modes
+- selstimset: select all stimset wave names
+- selivsccsetqc: IVSCC Set QC is ignored
+- selivsccsweepqc: IVSCC Sweep QC is ignored
+- selexp: experiment name is ignored
+- seldev: device name is ignored
+- selsetcyclecount: set cycle count is ignored
+- selsetsweepcount: set sweep count is ignored
+- selsciindex: stimset cycle id index is ignored
+- selracindex: repeated acquisition is index is ignored
+- selexpandrac: expansion by repeated acquisition cycle is disabled
+- selexpandsci: expansion by stimset cycle id is disabled
+
If select arguments appear multiple times then the resulting selection is an intersection of all sweep/channel combinations that were selected
+from all these select arguments.
+i.e. if one select argument has Sweep 0 AD0, Sweep 1 AD0 selected and a second select argument has Sweep 1 AD0 selected then
+only Sweep 1 AD0 remains selected because it appears in all selections.
+
The range specified through selrange is always taken from the topmost select.
+
If an experiment is specified with a wildcard pattern through selexp then there must be only a single matching experiment. The same applies for seldev. Only when the source is from a SweepBrowser with different loaded experiments then using selexp is senseful as e.g. for a DataBrowser the experiment is always the current experiment.
+
The filter criteria of the select filters are orthogonal (independent of each other) except for selsciindex, selracindex, selexpandrac and selexpandsci.
+Internally first the orthogonal select filters are applied. Then based on the resulting selections selsciindex, selracindex is applied, then selexpandsci, selexpandsci.
+Intersections with additional selections from select type arguments are executed afterwards.
+This implies that created selections can not be further filtered once created (see example).
+
When selexpandrac is used then the selected sweep numbers from selsweeps are extended. For each sweep selected by
+selsweeps sweeps numbers of the same repeated acquisition cycle are added. For the new sweep numbers selections are gathered with a modified copy of the initial selection filter:
+- selvis is changed to all
+- selexp is set to the experiment of the sweep number that was extended
+- seldev is set to the device of the sweep number that was extended
+
The resulting selections are gathered for each additional sweep number. Finally all selections are reduced to be unique only.
+
When selexpandsci is used then first the selection is retrieved for selsci disabled. Then for each selection
+for the sweep number / channel number / channel type combination the sweep numbers with the same stimset cycle id are determined. For these sweeps selections with the same channel number / channel type are added.
+Finally all selections are reduced to be unique only.
+
The expansion through selexpandsci and selexpandsci operates on the current select filter.
+
The output is composite with two datasets of different type.
+The first dataset contains a N x 4 array where the columns are sweep number, channel type, GUI channel number and row index of the sweepMap. The sweepMap only exists if the window is a SweepBrowser, for DataBrowser the values are set NaN in that column.
+The second dataset contains a dataset with range specification.
+
The output of the N x 4 array is sorted. The order is sweep -> channel type -> channel number.
+e.g. for two sweeps numbered 0, 1 that have channels AD0, AD1, DA6, DA7 from a DataBrowser:
+{{0, 0, 0, 0, 1, 1, 1, 1}, {0, 0, 1, 1, 0, 0, 1, 1}, {0, 1, 6, 7, 0, 1, 6, 7}, {NaN, NaN, NaN, NaN, NaN, NaN, NaN, NaN}}.
+
If the mode for selvis is displayed and no traces are displayed then a null wave is returned.
+If there are no matching sweeps found a null wave is returned.
# For sel2 the SCI expansion applies to sweep 0, AD0. The selection result
+# of the expansion is then intersected with sweep 1 AD0 that was selected
+# through sel1.
+sel1=select(selchannels(AD0),selsweeps(1),selvis(all))
+sel2=select(selchannels(AD0),selsweeps(0),selvis(all),selexpandsci(),$sel1)
+
+
+
# Note that the sel2 expression does not do a post-filtering of sel1
+# instead selracindex(5) is applied to the selections resulting from
+# the default filter setting for select for the case there is a select type argument present
+# Then these selections are intersected from sel1
+# Logically the intersection of the resulting selection works only for the orthogonal filter properties as a kind-of post-filter
+sel1=select(selsciindex(3))
+sel2=select(selracindex(5),$sel1)
+
The range function is borrowed from python. It expands
+values into a new array.
+
This function can also be used as an operation with the “…” operator which is
+the Unicode Character ‘HORIZONTAL ELLIPSIS’ (U+2026). Writing “…” is automatically converted to “…”.
The function generally accepts 1 to 3 arguments. The operation is intended to be
+used with two arguments.
+
The operation accepts also multiple data waves as first argument. Each data wave content must follow the operation argument order and size in that case.
+For this case the operation is applied on each input data wave independently and returns the same number of data waves.
+The returned data type is SF_DATATYPE_RANGE.
the name(s) of the epoch. The names can contain wildcard * and !.
+
+
selectData
+
the second argument is a selection of sweeps and channels where the epoch information is retrieved from. It must be specified through the select operation. Any range specification that is part of the select result is ignored.
When the optional second argument is omitted, select() is used as default that includes all displayed sweeps and channels.
+
+
+
+
type
sets what information is returned. Valid types are: range, name or treelevel. If type is not specified then range is used as default.
+
+
+
The operation returns for each selected sweep times matching epoch a data wave. The sweep meta data is transferred to the output data waves.
+If there was nothing selected the number of returned data waves is zero.
+If the selection contains channels that do not have epoch information stored these are skipped in the evaluation. For associated AD channels the epoch information is retrieved from the associated DA channel.
+If a selection has epoch information stored in the labnotebook and the specified epoch does not exist it is skipped and thus, not included in the output waves.
+
The output data varies depending on the requested type. Multiple epochs for one
+sweep always result in additional columns.
+
range:
+Each output data wave is numeric and has the start/end times in the rows [ms].
+
name:
+Each output data wave is textual and contains name of the epoch.
+
treelevel:
+Each output data wave is numeric and has the tree level of the epoch.
+
The returned data type is SF_DATATYPE_EPOCHS.
+The default suggested x-axis values for the formula plotter are sweep numbers. The suggested y-axis label is the combination of the requested type (name, tree level, range) and the epoch name wildcards.
The mode argument sets what test pulse analysis is run.
+The following tp analysis modes are supported:
+
tpbase() Returns the baseline level in pA or mV depending on the clamp mode.
+
tpinst() Returns the instantaneous resistance values in MΩ.
+
tpss() Returns the steady state resistance values in MΩ.
+
tpfit(stringfitFunc,stringretValue[,variablemaxTrail]) Returns results from fitting the test pulse range.
+
See specific subsections for more details.
+
The second argument is a selection of sweeps and channels where the test pulse information is retrieved from.
+It can be either a single select or an array with select`s. If an array of selects is specified then over each selection is iterated independently.
+If the optional second argument is omitted, `select() is used as default that includes all displayed sweeps and channels.
+Any range specification from the select is ignored when used with tp.
+The tp operation pre-filters the selected sweeps, only sweeps with channel type AD are used.
+
The optional argument ignoreTPs allows to ignore some of the found test-pulses. The indices are zero-based and identify the
+test-pulses by ascending starting time.
+
If a single sweep contains multiple test pulses then the data from the test
+pulses is averaged before analysis. The included test pulses in a single sweep must have the same duration.
+
The operation returns multiple data waves. There is one data wave returned for each sweep/channel selected through selectData.
+The sweep and channel meta data is included in each data wave.
+
The returned data type is SF_DATATYPE_TP.
+The default suggested x-axis values for the formula plotter are sweep numbers. The suggested y-axis label is the unit of the analysis value (pA, mV, MΩ).
+
Test pulses that are part of sweeps are identified through their respective epoch short name, that starts with “TP” or “U_TP”.
+If in selectData nothing is selected the number of returned data waves is zero.
+If a selected sweep does not contain any test pulse then for that data wave a null wave is returned.
+
# Get steady state resistance from all displayed sweeps and channels
+tp(tpss())
+
+# Get steady state resistance from all displayed sweeps and AD channels
+tp(tpss(),select(selchannels(AD)))
+
+# Get base line level from all displayed sweeps and AD1 channel
+tp(tpbase(),select(selchannels(AD1)))
+
+# Get base line level from all displayed sweeps with AD1 channel and all sweeps with AD2 channel
+tp(tpbase(),[select(selchannels(AD1)),select(selchannels(AD2),selvis(all))])
+
+# Get base line level from all displayed sweeps and channels ignoring test pulse 0 and 1
+tp(tpbase(),select(),[0,1])
+
+# Fit the test pulse from all displayed sweeps and channels exponentially and show the amplitude.
+tp(tpfit(exp,amp))
+
+# Fit the test pulse from all displayed sweeps and channels double-exponentially and show the smaller tau from the two exponentials.
+# The fitting range is changed from the default maximum of 250 ms to 500 ms if the next epoch is sufficiently long.
+tp(tpfit(doubleexp,tausmall,500))
+
The tpbase operation specifies an operation mode for the tp operation.
+In that mode the tp operation returns the baseline level in pA or mV depending on the clamp mode.
+tpbase uses a fixed algorithm and takes no arguments.
The tpss operation specifies an operation mode for the tp operation.
+In that mode the tp operation returns the steady state resistance values in MΩ.
+tpss uses a fixed algorithm and takes no arguments.
The tpinst operation specifies an operation mode for the tp operation.
+In that mode the tp operation returns the instantaneous resistance values in MΩ.
+tpinst uses a fixed algorithm and takes no arguments.
The tpfit operation specifies an operation mode for the tp operation.
+In that mode the tp operation fits data from test pulses with the specified fit function template and returns the specified fit result value.
+By default the fit range includes the epoch that follows after the test pulse limited up to 250 ms. Whichever ends first. The default time limit can be overwritten with
+the third argument.
The first argument is the name of a fit function, valid fit functions are exp and doubleexp.
+The fit function exp applies the fit: \(y = K_0+K_1*e^{-\frac{x-x_0}{K_2}}\).
+The fit function doubleexp applies the fit: \(y = K_0+K_1*e^{-\frac{x-x_0}{K_2}}+K_3*e^{-\frac{x-x_0}{K_4}}\).
+
The second argument specifies the value returned from the fit function. Options are tau, tausmall, amp, minabsamp and fitq.
+The option tau returns for the fit function exp the coefficient \(K_2\), for doubleexp it returns \(max(K_2, K_4)\).
+The option tausmall returns for the fit function exp the coefficient \(K_2\), for doubleexp it returns \(min(K_2, K_4)\).
+The option amp returns for the fit function exp the coefficient \(K_1\), for doubleexp it returns \(K_1\) if \(max(|K_1|, |K_3|) = |K_1|\), \(K_3\) otherwise.
+The option minabsamp returns for the fit function exp the coefficient \(K_1\), for doubleexp it returns \(K_1\) if \(min(|K_1|, |K_3|) = |K_1|\), \(K_3\) otherwise.
+The option fitq returns the fit quality defined as \(\sum_0^n{(y_i-y_{fit})^2}/(x_n-x_0)\).
+
The optional third argument specifies the time in [ms] after the test pulse that is included in the input data for the fit.
+The trail starts at the begin of the TP_B1 epoch. A maxTrail value of zero refers to the end of the TP_B1 epoch.
+The value of maxTrail can be negative up to the begin of TP_B1.
+If maxTrail is not set then the trail range ends at the beginning of the next epoch on tree level 1 or 250 ms after the end of TP_B1, whichever occurs first.
The log operation prints the first element of input wave to the command line but
+passes the wave transparently to the next operation. It is useful for debugging
+inside large formulas.
+
The operation accepts also multiple data waves as first argument.
+For this case the operation is applied on each input data wave independently and returns the same number of data waves.
+
If the input wave is empty, then log prints nothing and the number of data waves returned is zero.
The log10 operation applies the decadic (base 10) logarithm to its input.
+
The operation accepts also multiple data waves as first argument.
+For this case the operation is applied on each input data wave independently and returns the same number of data waves.
The store operation stores data in the labnotebook.
+
store(stringname,arraydata)
+
+
+
+
name
name suffix for the labnotebook entry. The full entry name is “Sweep Formula store [name]” without brackets.
+
+
data
a data wave.
+
+
+
The entries are written to the textual results wave for documentation purposes and
+later querying. The second parameter which can be any numerical/textual array,
+or output from other operations, is serialized and stored under the given name.
+
The operation returns the data argument unchanged.
+
store("fancy feature",[10,100])
+
+
+
adds the entry “Sweep Formula store [fancy feature]” with a serialized version
+of given array. The serialization format is JSON as described in the
+preliminary specification.
When clicking the Display button in the SF tab the formula gets parsed, executed and
+the result plotted. Evaluating the JSON object from the Formula Parser through the Formula Executor
+gives a resulting wave.
+For each data wave, the data from the rows is plotted as traces and the columns and layers
+are evaluated as an array of traces. Thus, a single plotted trace is created by the following input:
+1, 2, 3, 4, 5. Two traces with 5 data points each are created by this input:
+[1, 3], [2, 4], [3, 5], [4, 6], [5, 7]. Whereas the input 0…10, 20…30 creates
+ten traces with two data points each, starting with the first trace X = 0, Y = 0; X = 1, Y = 20.
+
In typical use cases instead of explicitly writing static data in the formula the data
+operation is utilized that returns data in the correct layout.
+
The plotter parses the meta data from data waves as well. For suitable data types
+trace colors and legend annotations are associated automatically. Operations can suggest x-values and x-axis labels to the plotter.
+If the user has not specified a formula for the x-values then the plotter uses the suggested x-values instead.
+
If the formula results returns a null wave as wave reference wave an error is generated by the formula plotter.
+If the formula results contains data waves that are null waves they are skipped by the formula plotter.
The formula plotter supports that the y-data or the x-data can be a 1d-text-wave. The other wave must be numeric.
+2d-text-waves are not supported for plotting.
Sometimes it is useful to explicitly specify X values for a series of data values.
+Therefore, two formulas can be plotted against each other by using the vs operator.
+
0...10vsrange(10,100,10)
+
+
+
gives
+
+
Note that in this example there are 10 Y-values and only 9 X-values returned by the
+respective formula part. The resulting graph shows 9 data points and thus does not show
+data points where either an X or Y value for the X, Y value pair is missing.
In the example the select operation selects channel AD0 from sweep 4, 5, 6, 7, 8, 9, 10 and 11. Thus, the data operation returns exactly 8 data waves with sweep data.
+Therefore, the min operation returns 8 data waves with exactly one data point. With the specified X-wave that also contains 8 points
+the first data wave from min gets the first value of the X-wave paired, the second data wave from min gets the second value of the X-wave paired a.s.o.
Several y-formulas can be plotted with the keyword with. The with must be
+on an own line between the y-formulas. If the y-data contains different data
+units the y-axis will show all data units separated by /. The vs allows to
+set a custom x-formula for the single y-formula left to it. Variables, see
+next section, can be used to reuse x-formulas for multiple statements without
+code duplication.
Variables store results of expressions. In formulas variables are included as strings prefixed by $.
+They are specified in the lines before the formula expression. The format of a variable definition is
+variableName = expression. The variable name must start with a letter. Further allowed letters are alphanumeric and _.
+The variable names are treated case-insensitive.
The section containing the variable definition can contain empty lines. The first line that is not fulfilling the format for a variable definition is treated as the first line
+of the formula expression(s) section. Variable definitions can use variables that were defined in a preceding line.
Previous variable content is discarded when the formula notebook is executed.
+
+
Limitations of the current variable definition concept:
+
The expression for a variable definition is resolved to a single wave reference wave
+
A single variable can not replace multiple arguments of an operation as operation arguments are processed one-at-a-time.
+
+
+
+
# This does NOT work
+c=cursors(A,B)
+s=select(selrange($c),selchannels(AD),selvis(all))
+p=$s,$s# p is resolved to a single numerical array
+
+data($p)# the data operation sees a single argument
+
+
+
As a general rule of thumb the result of an operation is a single wave reference wave and thus valid for a variable assignment.
+
Variables are stored in the Data/SweepBrowsers data folder in the variableStorage wave.
In the Sweep Formula notebook it is possible to get a quick help for operation and keywords like vs and and.
+Mark the operation in question with the mouse and hover over it, a tooltip appears that shows the help for this operation.
+Alternatively hold shift and right-click to jump to the Help tab that shows the help for the marked operation.
The following sketches some templates to write an operation.
+
Generally the JSON must not be parsed by the operation itself, but the wrapper functions have to be used.
+
Steps:
+
+
Get and check the number of arguments.
+
Retrieve and check all mandatory arguments. Use SF_GetArgument for arguments that can consist of multiple data waves and SF_GetArgumentSingle for arguments that are expected to return only a single data wave.
+
Retrieve all optional arguments from last to first and set for each a default value of not present. (see also operation code for setscale)
+
Create a output waveRef wave with SF_CreateSFRefWave of the correct size.
+
Execute the operation calculation, typically for each input data wave independently.
+
Be aware that a data wave might be a null wave, check sanity of input data wave, tranfer scales from input to calculation result if possible
+
Handle the Meta data, set a data type and transfer the wave notes on demand.
+
Return the operation result(s) through SF_GetOutputForExecutor or SF_GetOutputForExecutorSingle if the operation has only a single data wave as result.
+
Add the data type handling in SF_GetTraceColor and SF_GetMetaDataAnnotationText for proper trace colors and legend annotations in the formula plotter.
+
+
Example code for a typical operation taking three arguments, the first argument is some kind of input data.
+
staticFunction/WAVESF_OperationTemplate(variablejsonId,stringjsonPath,stringgraph)
+
+variablenumArgs
+stringinDataType
+
+numArgs=SF_GetNumberOfArguments(jsonID,jsonPath)// Get number of arguments 0 to N
+SF_ASSERT(numArgs<=3,"Operation has 3 arguments at most.")// Check if number of arguments is correct
+SF_ASSERT(numArgs>1,"Operation needs at least two arguments.")
+
+WAVE/WAVEarg0=SF_GetArgument(jsonID,jsonPath,graph,SF_OP_OPSHORTNAME,0)// Get first argument, this getter allows multiple data waves in the argument
+// For easy operation arguments it is good to have only a single argument with multiple data waves
+
+WAVEarg1=SF_GetArgumentSingle(jsonID,jsonPath,graph,SF_OP_OPSHORTNAME,1,checkExist=1)// Get second argument, only a single data wave is expected that must exist
+SF_ASSERT(DimSize(arg1,ROWS)==1,"Too many input values for argument two")// Sanity checks for second argument
+SF_ASSERT(IsNumericWave(arg1),"opName argument two must be numeric")
+
+// Parse optional arguments from last to first
+if(numArgs== 3)
+WAVEarg2=SF_GetArgumentSingle(jsonID,jsonPath,graph,SF_OP_OPSHORTNAME,2,checkExist=1)
+SF_ASSERT(DimSize(arg2,ROWS)==1,"Too many input values for parameter edge")
+SF_ASSERT(IsNumericWave(arg2),"edge parameter must be numeric")
+else
+// Set default value for optional argument if not existing
+Make/FREEedge={FINDLEVEL_EDGE_BOTH}
+endif
+
+// Create output wave
+WAVE/WAVEoutput=SF_CreateSFRefWave(graph,SF_OP_OPSHORTNAME,DimSize(arg0,ROWS))
+output=OperationCalculation(arg0[p],arg1[0],arg2[0])
+
+// Handle meta data
+// Set data type and transfer sweep information if input data was of the correct type
+
+SetStringInJSONWaveNote(results,SF_META_DATATYPE,SF_DATATYPE_THISOP)
+inDataType=GetStringFromJSONWaveNote(dataRef,SF_META_DATATYPE)
+if(!CmpStr(inDataType,SF_DATATYPE_SWEEP))
+SF_TransferFormulaDataWaveNote(arg0,output,"Sweeps",SF_META_SWEEPNO)
+endif
+
+// Return multiple data waves to executor, the function will wrap the wave ref wave to a one element text wave
+returnSF_GetOutputForExecutor(results,graph,SF_OP_OPSHORTNAME)
+End
+
+staticFunction/WAVEOperationCalculation(WAVE/Zinput,variablearg1,variablearg2)
+
+if(!WaveExists(input))
+return$""
+endif
+
+// Sanity checks on input data waves
+SF_ASSERT(IsNumericWave(input),"opname requires numeric data as input")
+SF_ASSERT(WaveDims(input)<=2,"opname accepts only upto 2d data")
+SF_ASSERT(DimSize(input,ROWS)>0,"opname requires at least one data point")
+// Do the actual calculation
+MatrixOP/FREEout=sqrt(averageCols(magsqr(input)))^t
+// Transfer the scaling if possible
+SF_FormulaWaveScaleTransfer(input,out,COLS,ROWS)
+
+returnout
+End
+
+
+
Example code for an operation taking the top array as input data. The specific difference here is that we use a convention that if there
+is only a single argument then we parse it as it could possibly an argument with multiple data waves. If it is just regular data then it is converted
+to a single data wave with one element and thus, stays compatible with the SF_GetArgumentTop parsing, if that would have encountered a single element.
+This allows to put output from e.g. data directly in such an operation as first argument. The operation works then on each data wave separately.
+
staticFunction/WAVESF_OperationTemplate(variablejsonId,stringjsonPath,stringgraph)
+
+variablenumArgs
+
+numArgs=SF_GetNumberOfArguments(jsonId,jsonPath)
+if(numArgs> 1)
+WAVE/WAVEinput=SF_GetArgumentTop(jsonId,jsonPath,graph,SF_OP_OPSHORTNAME)
+else
+WAVE/WAVEinput=SF_GetArgument(jsonId,jsonPath,graph,SF_OP_OPSHORTNAME, 0)
+endif
+WAVE/WAVEoutput=SF_CreateSFRefWave(graph,SF_OP_OPSHORTNAME,DimSize(input,ROWS))
+
+output[]=OperationCalculation(input[p])
+
+SFH_TransferFormulaDataWaveNoteAndMeta(input,output,SF_OP_OPSHORTNAME,SF_DATATYPE_OP)
+
+returnSF_GetOutputForExecutor(output,graph,SF_OP_OPSHORTNAME,clear=input)
+End
+
+staticFunction/WAVEOperationCalculation(WAVE/Zinput)
+
+// data waves can be null
+if(!WaveExists(input))
+return$""
+endif
+
+SF_ASSERT(IsNumericWave(input),"opName requires numeric input data.")
+SF_ASSERT(DimSize(input,ROWS)>0,"opName input must have at least one data point")
+// Do actual calculation
+WAVEout=NewFreeWave(IGOR_TYPE_64BIT_FLOAT, 0)
+Integrate/METH=1/DIM=(ROWS)input/D=out
+// Transfer scales and adapt
+CopyScalesinput,out
+SetScale/Px,DimOffset(input,ROWS),DimDelta(input,ROWS),"dx",out
+
+returnout
+End
+
+
+
The function SFH_TransferFormulaDataWaveNoteAndMeta transfers the meta information and wave notes of the reference and data waves.
+It also updates the operation stack information. There are two cases where SFH_TransferFormulaDataWaveNoteAndMeta can not be used:
+
+
The operation does not take an input reference wave
+
The operation returns data through SF_GetOutputForExecutorSingle that creates the reference wave.
+
+
For operations that do not take an input reference wave that is calculated to an output reference wave the approach is to update the operation stack
+meta information directly through JWN_SetStringInWaveNote(output, SF_META_OPSTACK, AddListItem(SF_OP_OPSHORT, “”)).
+If SF_GetOutputForExecutorSingle is called then the optional parameter opStack should be set to the previous operation stack. For operations like
+selsweeps() there is no previous operation, thus the parameter would be opStack=””.
+
It should be noted that there is a difference for parsing a single first argument through SF_GetArgument or SF_GetArgumentTop.
+SF_GetArgument starts execution for argument 0 specifically at the /0 JSON path location, whereas SF_GetArgumentTop starts execution at /.
+Set the case that the first argument is wave(pathToWave) with a 1d-wave containing a single element with value 17.
+SF_GetArgument executes the wave operation first, whereas SF_GetArgumentTop executes the array [wave(pathToWave)] first.
+Thus, SF_GetArgument sees with the resolved wave operation [17], whereas SF_GetArgumentTop sees [[17]]. Therefore the first returns a
+{17} wave and the latter a {{17}} wave.
+
More complex operation such as data build the output wave reference wave dynamically. See SF_GetSweepsForFormula how the output wave is build depending on selectData and the found sweeps.
Operation as well as the formula plotter can evaluate returned meta data from the result wave(s). Generally meta data is set through JSON wave notes.
+Data wave independent meta data is set in the wave ref wave, whereas data wave dependent data is set as note of the data wave(s) itself.
+Currently certain key constants for meta data fields are defined.
+
For the wave ref wave:
+
+
SF_META_DATATYPE: string, data type of operation result (some operations are transparent for that)
+
SF_META_XAXISLABEL: string, suggested label for the x-axis for the plotter, typically combined with x-value meta data in the data wave(s)
+
SF_META_YAXISLABEL: string, suggested label for the y-axis for the plotter
+
SF_META_OPSTACK: string, tracks the operation stack
+
+
For the data wave(s):
+
+
SF_META_SWEEPNO: number, number of the sweep that provided the source data
+
SF_META_CHANNELTYPE: number, channel type from the sweep that provided the source data
+
SF_META_CHANNELNUMBER: number, channel number from the sweep that provided the source data
+
SF_META_XVALUES: wave, suggested x-wave for the plotter to display this data wave
+
+
See also SF_OperationLabnotebookImpl, where such meta data is set.
+
The function SFH_TransferFormulaDataWaveNoteAndMeta transfers meta data from one operation to the next.
+If the following conditions are met then a suggested X-values are set in the meta data:
+
+
The input data type is SF_DATATYPE_SWEEP and all output data waves have no wave units for x set and all output data waves have only one data point -> sweep number is set as X-value and “Sweeps” as x label
+
For any not above specified input data type: if all output data wave have one data point and all output data waves have no wave units for x set and the input data wave has a sweep number value set in the meta data -> sweep number is set as X-value and “Sweeps” as x label
In the above example the data operation sets sweep number as meta data. The SFH_TransferFormulaDataWaveNoteAndMeta function transfers that meta data also to the results of the outer operations.
+The data waves returned from the min operation contain only a single data point and the result complies with the second set of conditions mentioned above. Thus, the results are
+displayed in the plotter with sweep numbers on the x-axis and “Sweeps” as x-label.
The operation stack meta data is updated in the called operation, typically through SFH_TransferFormulaDataWaveNoteAndMeta.
+It is a semicolon separated list of operations called for a single formula, where the most recent operation is at the front of the list.
+Operations where data from several sources is joined, like plus discard the previous operation stack. Thus, the operation stack contains
+only operations that were relevant for the strands of data that reaches ultimately the formula plotter.
+The operation stack information is used to create the trace legend(s) in the graph(s) as well as for the trace names.
+Also the trace color is determined through evaluation of the operation stack. For example, only if the operation stack indicates that the most recent data
+originated from a data() operation without intermediate operations that break this data strand, such as +, then the meta iformation about sweep data is taken to
+determine the traces color.
The idea of the argument setup stack is to store the arguments of each operation to be able determine differences between formulas
+in the end. This information can be used to change the trace style for differently setup formulas when plotted in the same graph with the
+with keyword. Also in the legend it can be shown what was setup differently.
+Operations can prepare argument setup information through a key/value style text wave with two columns. The wave is created with
+SFH_GetNewArgSetupWave and is filled then by the operation e.g.:
This information is stored when SFH_TransferFormulaDataWaveNoteAndMeta is called with the optional argSetup argument.
+If not setup by the operation, by default the only argSetup entry is the operation short name. Thus, the information content
+without setting it up is the same as in the operation stack.
+
The information is evaluated in the Formula Plotter to determine if traces from different formulas specified through the with keyword
+need to be shown with a different marker or line style. It also adapts the legend to show details about differences in arguments in formulas.
When using the psx SweepFormula operation the normal graphing is
+enhanced by various additional graphs and user interface options.
+The following graph shows the layout for a psx and psxStats operation with
+real acquired data.
+
+
The main panel SweepFormulaplotfrom<Browser> has a left vertical column
+with UI controls, the psx graph at the top and a psxstats graph at the bottom.
+
Above the main panel are again some UI controls on the very left, the all event
+graph and then the single event graph at the very right.
Store: Store the event properties including the classification results
+(EventState and FitState) in the results waves using the
+id as key and JSON as serialization format.
+
Load: Load the event properties from the results wave using the currently
+active id
+
Jump: Jump to the next event having event state undetermined, also
+changes the combination if required.
+
SuppressUpdate: Avoid updating the top all and single event graphs on
+event/fit state and/or cursor changes.
+
ListBox: The combinations listbox has an entry for every combination of
+range and select where psx found at least one event. The psx
+graph only ever plots one combination, and therefore one can choose the
+to-be-plotted combination here.
+
i: Button to copy the psx/psxkernel/psxRiseTime input parameters to
+the clipboard
+
UI: Hover over this element with the mouse to learn about the keyboard shortcuts
The psx graph represents the results of the event search and is split
+vertically into two graphs. The top one shows the filtered and offsetted sweep
+data, the lower one in addition the deconvolution with the input from psxkernel.
+
All found events are marked initially as undetermined for the Event/Fit state.
+
The Event state is visualized by colorized markers: a grey square for
+undetermined, a green circle for accept and a red triangle for reject.
+
With the A cursor a single event can be selected, properties of that event are
+displayed in the single event graph (top right).
+
The following options exist to set the event and fit state of a single or multiple events:
The following keyboard shortcuts work for either the psx or the psxstats
+graphs. All of them require that the cursor A is located on an event, which
+is by default the case for the psx graph. The current direction for automatic
+advancement defaults to left-to-right.
+
+
↑ (up): Accept the current event, changing both event and fit state to
+accept, and advance the cursor to the next event in the current direction
+
↓ (down): Reject the current event, changing both event and fit state to
+reject, and advance the cursor to the next event in the current direction
+
→ (right): Move the cursor to the next event on the right
+
← (left): Move the cursor to the previous event on the left
+
(space): Toggle the event and fit state of the current event without any movement
+
r: Reverse the current direction
+
c: Center the x-axis around the current event
+
e: Toggle the event state
+
f: Toggle the fit state
+
z: Accept the event state but reject the fit state
The combination to select the drawn single events and their averages from are
+determined by the state of the Currentcombo checkbox. The state to use for
+the selection, either fit or event state, is determined by the popup menu.
+
+
Accept: Show all accepted events
+
Reject: Show all rejected events
+
Undetermined: Show all undetermined events
+
Average:
+
+
Accept: Show all the average of all accepted events
+
Reject: Show all the average of all rejected events
+
Undetermined: Show the average of all undetermined events
+
All: Show the average of all events regardless of the state
+
+
Fit: Show the fit of the averaged events, the fit coefficients are also
+
stored in the results wave
+
+
+
i: Hover over this element with the mouse to view the average fit
+coefficients
+
+
+
+
+
+
+
Currentcombo: Restrict the event selection to the currently selected
+combination (checked) or use all of them (unchecked)
+
FitState/EventState: Select the state to use as basis for selection
+
dblexp_peak/dblexp_XOffset: Select the fit curve for the average fit
+
Onset/peak: Select the event property to offset the events in the all
+events graph to
+
Blocksize[%]: Percentage to select what part of the events are
+displayed in the all events graph. This can help with reducing the number of
+plotted events so that the interactive event classification in the all event
+graph is easier.
+
Block: Block number to display in the all events graph in case a block
+size smaller than 100% was selected.
The top left graph shows events and average selected by the UI controls to its
+left. The state can be changed by hovering the mouse over an event trace and
+pressing the keyboard shortcuts available for the psx graph.
The top right graph shows a single event including its fit (greyish line) which might
+not be present as well. The shown event is selected by cursor A in the psx
+graph. The textbox has various properties of the shown event.
The algorithm analyses the measured test pulse response to determine the instantaneousresistance,
+the steadystateresistance and the baselinesteadystate.
+
The TP_Delta() function is called after data is available for each device.
+This data can contain test pulse responses from several head stages from multiple ADC channels.
The goal of the algorithm is to determine the base line level,
+steady state resistance and instantaneous resistance of a test pulse response.
+The available data contains the response pulse as
+well as information about the excitation amplitude and length.
+The excitation pulse is in units of current or voltage depending on the used
+clamp mode. The actual data consists of discrete points of analog to digital
+input (response).
+For simplicity this will be neglected in the following.
+
The resistance is calculated by
+
+\[R = \frac{U}{I}\]
+
Where depending on voltage- or current clamp mode the response test pulse is either
+a current or voltage. The response is the difference between base line live and
+steady state / instantaneous level. For determining the levels in the response data
+ranges are defined where the data points get averaged.
Where baselinefrac is the part relative to the full test pulse range in the
+front of the test pulse where the excitation signal was at its base line level.
+The time duration is the length of the active portion of the exciting test pulse.
+
The range for averaging for the base line level is defined minimum of:
The range for averaging for the steady state level is defined to be equal to the
+base line averaging length. The reference point for the end of the range is
+defined to be close to the end of the active test pulse:
The range for averaging for the instantaneous level is 0.25 ms. The reference
+point for the start of the range is defined to be close after the start of the
+active test pulse:
The device specific data folder for the test pulse is retrieved. The current and
+voltage clamp parameters are retrieved from it. It is used as well to put the
+calculated BaselineSSAvg, SSResistance and InstResistance back to the
+devices test pulse data folder.
+
DFREFdfr=GetDeviceTestPulse(device)
+
+
+
The actual test pulse data is retrieved from OscilloscopeData, where the data
+points are stored in rows and the columns count the DAC, ADC and TTL channels
+(in this order).
Retrieve device specific Current Clamp and Voltage Clamp amplitudes. The values
+are in pA and mV and can be set on the front panel in the tab
+“Data Acquisition”. Default: -50 pA / 10 mV.
+
Retrieve the column of the first ADC channel in OscilloscopeData wave,
+due to the DAC, ADC, TTL order it is 1 for one enabled head stage,
+2 for two enabled head stages a.s.o.
Retrieve head stage properties, where rows count the active head stages and
+columns enumerate the properties. It is used later to decide if a certain head
+stage operates in current clamp or voltage clamp mode.
+
WAVEactiveHSProp=GetActiveHSProperties(device)
+
+
+
The test pulse is centered on a baseline, the baselineFrac is a number < 1, that
+defines the fraction in front and after the test pulse. Example: With a typical
+value of 0.25 for baselineFrac, the whole test pulse consists of parts of
+0.25_baseline + 0.5_testpulse + 0.25_baseline.
+
Length of the buffer that stores previous results of BaselineSSAvg,
+SSResistance and InstResistance for a running average. The running
+average is later applied by TP_CalculateAverage() if the size is > 1.
+The size is set on the front panel in the Settings tab.
+
The later resistance calculation is based on R = U / I. Since R is always
+positive, the sign of the local clamp current/voltage variables is removed.
The length in time of the base line fraction is calculated by the fraction of the
+full test pulse length multiplied by the scale delta of the OscilloscopeData
+waves rows, which is the sample interval.
For the determination of the baseline level and the steady state level a small
+range of points is taken into account. The range the lowest value of either
+
+
+
5 ms
+
20% of the test pulse duration
+
20% of the base line duration
+
+
+
The range is converted to points by dividing through the sample interval.
The reference point for the base line determination is defined by the base line
+fraction multiplied by the length of the test pulse in points, which gives the
+onset point of the active test pulse.
The reference point for the steady state level determination is defined by
+1 - base line fraction multiplied by the length of the test pulse, which gives
+the end point of the active test pulse.
The range for the points to calculate the instantaneous resistance is a fixed range
+of 0.25 ms. It is converted to points by dividing the sample interval.
The reference point is defined by the base line
+fraction multiplied by the length of the test pulse in points, which gives the
+onset point of the active test pulse.
The steady state ranges are summed by columns (n x m to 1 x m wave) and divided
+the number of rows (i.e. number of points) to get the average per channel. The
+resulting wave is AvgTPSS (1 x m) holding the steady state averages.
The base line ranges are summed by columns (n x m to 1 x m wave) and divided
+by the number of rows (equals number of points per channel) to get the average
+per channel. The resulting wave is AvgBaselineSS (1 x m) holding the base
+line averages.
The base line average wave is duplicated to a reduced wave containing only the
+active ADC channels and is put back into the test pulse folder. A reference to
+the reduced wave is kept as BaselineSSAvg. The current number of AD channels
+is used to skip the DAC channels. The full remaining range is duplicated, requiring
+that no TTL channels are active (original column order from OscilloscopeData
+wave with DAC, ADC, TTL)
The absolute difference of the steady state level and the base line level is
+calculated by abs(AvgTPSS - AvgBaselineSS) per AD channel and stored in
+AvgDeltaSS.
A free wave InstAvg (1 x col) for calculating the instantaneous average is
+created, where col is the column number of OscilloscopeData, but at least 1.
The column of Instantaneous is extracted to a 1d
+free wave Instantaneous1d. WaveStats is applied to retrieve the point location
+of the minimum and maximum value V_minRowLoc and V_maxRowLoc in Instantaneous1d.
+The base line level average for the current AD channel is read from AvgBaselineSS
+to the variable OndDBaseline, which is not further used.
+Depending on the set clamp mode of the current AD channel from activeHSProp
+and the sign of V/I-clamp amplitude of the current device the maximum or minimum
+region is averaged:
+
+
+
V-clamp mode and positive amplitude -> maximum region
+
V-clamp mode and negative amplitude -> minimum region
+
I-clamp mode and positive amplitude -> maximum region
+
I-clamp mode and negative amplitude -> minimum region
+
+
+
The average is calculated by using the mean function that averages from scaled
+location x1 to x2. x1 is the scaled location for the point at V_maxRowLoc-1
+and x2 is the scaled location for the point at V_maxRowLoc+1. This effectively
+calculated the mean from three consecutive points in Instantaneous1d and
+puts it into the first row of InstAvg at the unreduced column position of
+the active AD channel.
+
The same averaging is done when the minimum region is targeted with V_minRowLoc.
+
The MultiThread directive is questionable here as it is a single value assignment.
+
+
do
+matrixOp/FreeInstantaneous1d=col(Instantaneous,i+ADChannelToMonitor)
+WaveStats/Q/M=1Instantaneous1d
+OndDBaseline=AvgBaselineSS[0][i+ADChannelToMonitor]
+if((activeHSProp[i][%ClampMode]==V_CLAMP_MODE?sign(amplitudeVCGlobal):sign(amplitudeICGlobal))==1)// handles positive or negative TPs
+MultithreadInstAvg[0][i+ADChannelToMonitor]=mean(Instantaneous1d,pnt2x(Instantaneous1d,V_maxRowLoc-1),pnt2x(Instantaneous1d,V_maxRowLoc+ 1))
+else
+MultithreadInstAvg[0][i+ADChannelToMonitor]=mean(Instantaneous1d,pnt2x(Instantaneous1d,V_minRowLoc-1),pnt2x(Instantaneous1d,V_minRowLoc+ 1))
+endif
+i+= 1
+while(i<(columnsInWave-ADChannelToMonitor))
+
+
+
Afterwards the absolute difference to the base line averages from
+AvgBaselineSS is calculated and put back to InstAvg. Also here the
+MultiThread is questionable as the wave is (1 x m) with m the number of channels.
The steady state delta wave is duplicated to a reduced wave containing only the
+active AD channels and is put back into the test pulse folder. A reference to
+the reduced wave is kept as SSResistance. The current number of AD channels
+is used to skip the DAC channels. The full remaining range is duplicated, requiring
+that no TTL channels are active (original column order from OscilloscopeData
+wave with DAC, ADC, TTL)
+
The x scale of the SSResistance wave is set to the time where the
+TPSSEndPoint is located. As SSResistance is a (1 x m) wave, this sets the
+time point for the averaged data.
+
Duplicate/O/R=[][ADChannelToMonitor,dimsize(TPSS,1)-1]AvgDeltaSSdfr:SSResistance/Wave=SSResistance
+SetScale/PxIndexToScale(OscilloscopeData,TPSSEndPoint,ROWS),1,"ms",SSResistance// this line determines where the value sit on the bottom axis of the oscilloscope
+
+
+
The instantaneous average wave is duplicated to a reduced wave containing only the
+active AD channels and is put back into the test pulse folder. A reference to
+the reduced wave is kept as InstResistance. The current number of AD channels
+is used to skip the DAC channels. The full remaining range is duplicated, requiring
+that no TTL channels are active (original column order from OscilloscopeData
+wave with DAC, ADC, TTL)
+
The x scale of the InstResistance wave is set to the time where the
+TPInstantaneousOnsetPoint is located. As InstResistance is a (1 x m) wave,
+this sets the time point for the averaged data.
For each active AD channel: The actual resistance is calculated by the formula R = U / I for
+SSResistance and InstResistance.
+
For I-clamp mode of the current channel:
+
+
+
SSResistance = AvgDeltaSS / amplitudeIC * 1000
+
InstResistance = InstAvg / amplitudeIC * 1000
+
+
+
For V-clamp mode of the current channel:
+
+
+
SSResistance = amplitudeVC / AvgDeltaSS * 1000
+
InstResistance = amplitudeVC / InstAvg * 1000
+
+
+
AvgDeltaSS contains the absolute difference of the steady state level and the base line level.
+InstAvg contains the absolute difference of the instantaneous level and the base line level.
+By default the current values are in pA and the voltage values in mV, thus the resulting resistance values are
+in MΩ.
+
i= 0
+do
+if(activeHSProp[i][%ClampMode]==I_CLAMP_MODE)
+// R = V / I
+MultithreadSSResistance[0][i]=(AvgDeltaSS[0][i+ADChannelToMonitor]/(amplitudeIC)) * 1000
+MultithreadInstResistance[0][i]=(InstAvg[0][i+ADChannelToMonitor]/(amplitudeIC)) * 1000
+else
+MultithreadSSResistance[0][i]=((amplitudeVC)/AvgDeltaSS[0][i+ADChannelToMonitor]) * 1000
+MultithreadInstResistance[0][i]=((amplitudeVC)/InstAvg[0][i+ADChannelToMonitor]) * 1000
+endif
+i+= 1
+while(i<(dimsize(AvgDeltaSS,1)-ADChannelToMonitor))
+
+
+
columns is set that holds the number of active AD channels, but at least 1.
+It is later used to set the number of ADCs for calling TP_RecordTP().
A running average is applied if tpBufferSize is greater than one.
+TPBaselineBuffer, TPInstBuffer and TPSSBuffer are the waves holding
+the values for the running average and are at maximum (tpBufferSize x m) in size.
+TP_CalculateAverage() takes the new value from the second parameter and puts the
+averaged value back therein.
+
if(tpBufferSize> 1)
+// the first row will hold the value of the most recent TP,
+// the waves will be averaged and the value will be passed into what was storing the data for the most recent TP
+WAVE/SDFR=dfrTPBaselineBuffer,TPInstBuffer,TPSSBuffer
+
+TP_CalculateAverage(TPBaselineBuffer,BaselineSSAvg)
+TP_CalculateAverage(TPInstBuffer,InstResistance)
+TP_CalculateAverage(TPSSBuffer,SSResistance)
+endif
+
numADCs is set to the value of columns and TP_RecordTP() is called to
+set the TPStorage wave with the given averages. All the input waves are reduced
+waves holding only AD channels. DQ_ApplyAutoBias() is called with the
+current values of BaselineSSAvg and SSResistance.
+Finally the elapsed time since function start is printed to the debug output.
Minimum absolute frequency distance for DAScale estimation. Only for “AdaptiveSupra”.
+
+
AbsoluteVoltageDiff
+
variable
+
required
+
VM
+
+
Maximum absolute allowed difference of the baseline membrane potentials [mV].
Set to inf to disable this check.
+
+
+
+
+
AmpBesselFilter
+
variable
+
optional
+
CR
+
+
Applies a bessel filter to the primary output.
Defaults to 10e3 [Hz],pass “100000” to select “Bypass”.
+
+
+
+
+
AmpBesselFilterRestore
+
variable
+
optional
+
CR
+
Restores the previously active bessel filter in POST_SET_EVENT. Defaults to ON.
+
+
AsyncQCChannels
+
wave
+
required
+
AR, CR, DA, PB, RA, RB, SE, SP, VM
+
List of asynchronous channels with alarm enabled, which must not be in alarm state.
+
+
AutoBiasBaselineModifier
+
variable
+
required
+
SC
+
Auto bias modifier value in mV on failing baseline QC.
+
+
AutobiasTargetV
+
variable
+
optional
+
CR
+
Autobias targetV [mV] value set in PRE_SET_EVENT
+
+
AutobiasTargetVAtSetEnd
+
variable
+
optional
+
CR
+
Autobias targetV [mV] value set in POST_SET_EVENT (only set if set QC passes).
+
+
BaselineChunkLength
+
variable
+
optional
+
AR, SE, VM
+
Length of a baseline QC chunk to evaluate (defaults to 500)
+
+
BaselineRMSLongThreshold
+
variable
+
optional
+
AR, CR, DA, PB, RA, RB, SE, VM
+
Threshold value in mV for the long RMS baseline QC check (defaults to 0.5)
+
+
BaselineRMSShortThreshold
+
variable
+
optional
+
AR, CR, DA, PB, RA, RB, SE, VM
+
Threshold value in mV for the short RMS baseline QC check (defaults to 0.07)
+
+
BaselineTargetVThreshold
+
variable
+
optional
+
CR, DA, RA, RB
+
Threshold value in mV for the target V baseline QC check (defaults to 1)
+
+
BoundsEvaluationMode
+
string
+
required
+
CR
+
Select the bounds evaluation mode: Symmetric (Lower and Upper), Depolarized (Upper) or Hyperpolarized (Lower)
+
+
DAScaleModifier
+
variable
+
mixed
+
CR, DA, SC
+
Modifier value to the DA Scale of headstages with spikes during chirp
+Percentage how the DAScale value is adapted if it is outside of the MinimumSpikeCount”/”MaximumSpikeCount” band. Only for “Supra”.
+Modifier value to the DA Scale of headstages for too few spikes
+
+
DAScaleOperator
+
string
+
mixed
+
CR, SC
+
Set the math operator to use for combining the DAScale and the modifier. Valid strings are “+” (addition) and “*” (multiplication).
+Set the math operator to use for combining the DAScale and the too few spikes modifier. Valid strings are “+” (addition) and “*” (multiplication).
+
+
DAScaleRangeFactor
+
variable
+
optional
+
DA
+
Fallback factor to apply to the input DAScale range for calculating the DAScale minimum step width when all previous f-I data from Rheobase/Supra/Adaptive is constant. Only for “AdaptiveSupra”.
+
+
DAScales
+
wave
+
required
+
DA, DS
+
DA Scale Factors in pA
+
+
DAScaleSpikePositionModifier
+
variable
+
required
+
SC
+
Modifier value to the DA Scale of headstages with failing spike positions.
+
+
DAScaleSpikePositionOperator
+
string
+
required
+
SC
+
Set the math operator to use for combining the DAScale and the spike position modifier. Valid strings are “+” (addition) and “*” (multiplication).
+
+
DaScaleStepWidthMinMaxRatio
+
variable
+
optional
+
DA
+
Number of times the maximum DAScale step width is larger than the calculated minimum. Only for “AdaptiveSupra”.
+
+
DaScaleTooManySpikesModifier
+
variable
+
required
+
SC
+
Modifier value to the DA Scale of headstages for too many spikes
+
+
DaScaleTooManySpikesOperator
+
string
+
required
+
SC
+
Set the math operator to use for combining the DAScale and the too many spikes modifier. Valid strings are “+” (addition) and “*” (multiplication).
+
+
FailedLevel
+
variable
+
mixed
+
CR, VM
+
Absolute level for spike search
+
+
FailedPulseLevel
+
variable
+
optional
+
SC
+
[Optional, uses the already set value] Numeric level to use for the failed pulse search in the PA plot tab.
+
+
FinalSlopePercent
+
variable
+
optional
+
DA
+
As additional passing criteria the slope of the f-I plot must be larger than this value. Note: The slope is used in percent. Only for “Supra”.
+
+
IdealNumberOfSpikesPerPulse
+
variable
+
required
+
SC
+
Ideal number of spike which should be present. Overwrites the PA plot GUI value.
+
+
InnerRelativeBound
+
variable
+
required
+
CR
+
Lower bound of a confidence band for the acquired data relative to the pre pulse baseline in mV. Must be positive.
+
+
InterTrialInterval
+
variable
+
optional
+
VM
+
Inter trial interval to set, defaults to 10 [s]
+
+
MaxAccessResistance
+
variable
+
required
+
AR
+
Maximum allowed acccess resistance [MΩ]
+
+
MaxAccessToSteadyStateResistanceRatio
+
variable
+
required
+
AR
+
Maximum allowed ratio of access to steady state resistance [%]
+
+
MaxFrequencyChangePercent
+
variable
+
optional
+
DA
+
The maximum allowed difference for the frequency for two consecutive measurements. In case this value is overshot, we redo the measurement with a fitting DAScale in-between. Only for “AdaptiveSupra”.
+
+
MaximumDAScale
+
variable
+
required
+
FR
+
Maximum allowed DAScale, set to NaN to turn the check off [pA]
+
+
MaximumSpikeCount
+
variable
+
optional
+
DA
+
The upper limit of the number of spikes. Only for “Supra”.
+
+
MaxLeakCurrent
+
variable
+
required
+
AR, PB
+
Maximum current [pA] which is allowed in the pre pulse baseline
+
+
MaxPipetteResistance
+
variable
+
required
+
PB
+
Maximum allowed pipette resistance [MΩ]
+
+
MaxTrials
+
variable
+
optional
+
SC
+
[Optional, defaults to infinity] A sweep is rerun this many times on a failed headstage.
+
+
MinimumSpikeCount
+
variable
+
optional
+
DA
+
The lower limit of the number of spikes. Only for “Supra”.
+
+
MinimumSpikePosition
+
variable
+
required
+
SC
+
Minimum allowed spike positions in pulse active coordinate system (0 - 100).
+
+
MinPipetteResistance
+
variable
+
required
+
PB
+
Minimum allowed pipette resistance [MΩ]
+
+
NextIndexingEndStimSetName
+
string
+
optional
+
AR, PB, SE, VM
+
+
Next indexing end stimulus set which should be set in case of success.
Also enables indexing.
+
+
+
+
+
NextStimSetName
+
string
+
optional
+
AR, PB, SE, VM
+
Next stimulus set which should be set in case of success
+
+
NumberOfChirpCycles
+
variable
+
optional
+
CR
+
Number of acquired chirp cycles before the bounds evaluation starts. Defaults to 1.
+
+
NumberOfFailedSweeps
+
variable
+
required
+
AR, CR, PB, SE, VM
+
Number of failed sweeps which marks the set as failed.
+
+
NumberOfSpikes
+
variable
+
required
+
RA
+
Number of spikes required to be found after the pulse onset in order to label the cell as having “spiked”.
+
+
NumberOfTestpulses
+
variable
+
required
+
AR, PB
+
Expected number of testpulses in the stimset
+
+
NumInvalidSlopeSweepsAllowed
+
variable
+
optional
+
DA
+
Number of allowed sweeps with invalid f-I fit slope. Only for “AdaptiveSupra”.
+
+
NumSweepsWithSaturation
+
variable
+
optional
+
DA
+
Number of required consecutive passing sweeps to have passing sweep QC and f-I slope QC, defaults to 2. Only for “AdaptiveSupra”.
+
+
OffsetOperator
+
string
+
optional
+
DA
+
Set the math operator to use for combining the rheobase DAScale value from the previous run and the DAScales values. Valid strings are “+” (addition) and “*” (multiplication). Only for “Supra” and defaults to “+”.
+
+
OperationMode
+
string
+
required
+
DA
+
Operation mode of the analysis function. Can be either “Sub”/”Supra”/”AdaptiveSupra”.
+
+
OuterRelativeBound
+
variable
+
required
+
CR
+
Upper bound of a confidence band for the acquired data relative to the pre pulse baseline in mV. Must be positive.
+
+
PostDAQDAScale
+
variable
+
required
+
FR
+
If true the found DAScale value will be set at the end of the set
+
+
PostDAQDAScaleFactor
+
variable
+
required
+
FR
+
Scaling factor for setting the DAScale of passed headstages at the end of the set
+
+
PostDAQDAScaleForFailedHS
+
variable
+
required
+
FR
+
Failed headstages will be set to this DAScale value [pA] in the post set event
+
+
PostDAQDAScaleMinOffset
+
variable
+
required
+
FR
+
Mininum absolute offset value applied to the found DAScale [pA]
+
+
RelativeVoltageDiff
+
variable
+
required
+
VM
+
+
Maximum relative allowed difference of the baseline membrane potentials [%].
Set to inf to disable this check.
+
+
+
+
+
SamplingFrequency
+
variable
+
optional
+
AR, CR, DA, PB, RA, RB, SE, SP, VM
+
Required sampling frequency for the acquired data [kHz]. Defaults to ITC:50 NI:50 Sutter:50.
+Required sampling frequency for the acquired data [kHz]. Defaults to ITC:200 NI:250 Sutter:50.
+Required sampling frequency for the acquired data [kHz]. Defaults to ITC:50 NI:50 Sutter:50.
+Required sampling frequency for the acquired data [kHz]. Defaults to ITC:50 NI:50 Sutter:50.
+Required sampling frequency for the acquired data [kHz]. Defaults to ITC:200 NI:250 Sutter:50.
+Required sampling frequency for the acquired data [kHz]. Defaults to ITC:50 NI:50 Sutter:50.
+Required sampling frequency for the acquired data [kHz]. Defaults to ITC:200 NI:250 Sutter:50.
+
+
SamplingMultiplier
+
variable
+
required
+
AR, CR, DA, FR, PB, RA, RB, SE, SP, VM
+
Sampling multiplier, use 1 for no multiplier
+
+
SealThreshold
+
variable
+
required
+
SE
+
Minimum required seal threshold [GΩ]
+
+
ShowPlot
+
variable
+
optional
+
DA
+
Show the resistance (“Sub”) or the f-I (“Supra”) plot, defaults to true.
+
+
SlopePercentage
+
variable
+
optional
+
DA
+
Once the slope of the f-I fit is x percentage smaller compared with the maximum slope, the stimset cycle is stopped. Only for “AdaptiveSupra”.
+
+
SpikeCheck
+
variable
+
optional
+
CR
+
Toggle spike check during the chirp. Defaults to off.
+
+
SpikeFailureIgnoredTime
+
variable
+
required
+
VM
+
+
Time [ms] to ignore around spikes for average calculation.
Half of this value before and after each spike.
+
+
+
+
+
TestPulseGroupSelector
+
string
+
optional
+
SE
+
Group(s) which have their resistance evaluated: One of Both/First/Second, defaults to Both
+
+
UserOffsetTargetVAutobias
+
variable
+
required
+
VM
+
Offset value for the autobias target voltage in case spikes were found
+
+
UserOnsetDelay
+
variable
+
optional
+
CR
+
Will be set as user onset delay in PRE_SET_EVENT, the current value will be set back in POST_SET_EVENT.
+
+
UseTrueRestingMembranePotentialVoltage
+
variable
+
optional
+
CR
+
Use the average voltage of a passing True RMS voltage set as Autobias targetV [mV] instead of “AutobiasTargetVAtSetEnd” in POST_SET_EVENT. Defaults to on.