[TUTORIAL UNDER DEVELOPMENT: NOT READY FOR PUBLIC USE]
Authors: Francois Tadel, Elizabeth Bock, Sylvain Baillet
The previous tutorials explained how to use Brainstorm in an interactive way to process one subject with two acquisition runs. In the context of a typical neuroimaging study, you may have tens or hundreds of subjects to process in the same way, it is unrealistic to do everything manually. Some parts of the analysis can be processed in batches with no direct supervision, others require more attention. This tutorial introduces tools and tricks that will help you assemble an efficient analysis pipeline.
Requirements: You need a license for the Matlab environment in order to use these tools and execute custom scripts. If you are running the compiled version of Brainstorm with the MCR library, the only custom code you can run is through the menu File > Matlab console and the process "Run Matlab command".
The easiest way to get started with a new Brainstorm script is to use the script generator, already introduced in the tutorial Select files and run processes. Select some files in the Process1 or Process2 tabs, select a list of processes, and use the menu Generate .m script. The example below should work on the the protocol "TutorialIntroduction" created during the introduction tutorials.
In the Process1 tab, leave the selection box empty and click on [Run]. Instead of selecting the files from the Brainstorm interface, we will select them directly from the database using a script.
Select process File > Select files: Recordings (do not execute immediately)
Subject=Subject01, Condition=[Empty], File comment=Avg: deviant (the space is important).
This process selects all the recordings with a comment including the string "Avg: deviant", from all the folders in Subject01 (except "Intra-subject" and "Common files"). We expect to get two files: the averages of the deviant condition for both runs.
Add process Pre-process > Band-pass filter: Lower cutoff=0Hz, Upper cutoff=30Hz, Mirror. Add process File > Save snapshot: Recordings time series, Sensors=MEG.
This will apply a low-pass filter at 30Hz and save a screen capture of the signals in the report.
Do not run the pipeline, select the menu Generate .m script instead. It saves a new .m file and opens it in the Matlab editor. Close the pipeline editor window and look at the script.
The script you just generated can be the starting point to your own custom script. The following sections explain line by line how they work and how to edit them.
Line by line: Header
% Script generated by Brainstorm (19-Jul-2016)
All the lines starting with a "%" are comments, they are never executed.
sFiles: The list of files in input. Currently empty because we did not select anything in input in the Process1 list. If we had selected files, it would contain a cell array of strings with relative file paths.
SubjectNames: List of subject names that are used in the script. Most of the times, the generated script would contain only one entry, but it written as a cell array so that it is easier to extend it to multiple subjects with a loop (described further in this tutorial).
% Start a new report
bst_report('Start', sFiles);
Start a new report of activity: Clears all the previous logs and gets ready to record new messages. The report will collect all the messages that are generated during the execution of the script by the various processes. You can explicitely add screen captures and additional messages to the current report with the function bst_report. This report will remain open until the function bst_report('Start') is called again. To display the current report, use the menu File > Report viewer.
The syntax function_name('SubFunction', arguments) is used a lot in Brainstorm, to call a subfunction available inside a .m file. This line above calls the function Report() in the file brainstorm3/toolbox/process/bst_report.m. This is made possible with the use of the short script "macro_methodcall" you will see at the top of many files. Many of the Brainstorm .m files are actually libraries of functions, rather than simple "scripts" or "functions".
You will find one block per process you selected in the pipeline editor. They all have the same syntax:
output_files = bst_process('CallProcess', process_name, input_files_A, input_files_B, options_list);
process_name: String indicating the function corresponding to the process to execute. To know from the pipeline editor what is the path to the process function: hover your mouse over the selected process, as illustrated in this tutorial.
input_files_A: List of input files in Process1, or FilesA in Process2. It can be a cell array of files names (full path, or relative path from the protocol folder), or an array of structures describing the files in the database (returned by a previous call to bst_process).
input_files_B: Empty for Process1, or FilesB in Process2. Cell array of strings or array of struct.
options_list: Pairs of (option_name, option_values), one for each option of the process.
output_files: Array of structures describing the files in output of the process. If the process created new files, this variable contains the new files. If the process didn't create new files or was modifying exiting files, most of the time this variable would contain the same files as the input list.
Line by line: Footer
% Save and display report
ReportFile = bst_report('Save', sFiles);
Closes the current report and saves it in the user Brainstorm report folder ($HOME/.brainstorm/reports). These reports are in .mat format and contain all the information necessary to re-run the execution exactly in the same way, but they not easy to read.
The parameter "sFiles" is optional, it indicates what are the files that are considered as the final results of the script. You can remove it without breaking your script: ReportFile = bst_report('Save');
bst_report('Open', ReportFile);
Opens the report viewer to display what happened during the execution. This is equivalent to using the menu File > Report viewer. You can comment this line (ie. add a "%" at the beginning of the line) if you don't want to show the report at the end of the execution.
% bst_report('Export', ReportFile, ExportDir);
This function exports the report in readable format, as an HTML that includes all the screen captures embedded in it. It is disabled by default. If you want to use this feature: remove the "%" at the beginning of the line, and define the variable ExportDir.
ExportDir must be a string that defines where to save the HTML report. It can be wither the absolute path to a HTML file (eg. 'C:\Users\myuser\Documents\report_example.html') or just a folder (eg. 'C:\Users\myuser\Documents'). If you enter only a path to a folder, a default file name including the protocol name and a date tag is generated (report_ProtocolName_YYMMDD_HHMMSS.html).
Simplify the calls
This script you generated is like any Matlab script: you can edit it, rename the variables, add tests and loops, etc. The first important thing to understand is how to edit the options or change the inputs/outputs of a process. The script generator uses only one variable for all the file lists (sFiles) and the output process is always the input of the following process. This is usually too restrictive to write a full analysis script: we commonly need to have multiple lists of files or to run two different operations on the same file.
Let's consider the first process call, which selects the averages for the Deviant condition in both runs.
There is no need to set the parameter sFiles, because there is no input, you can replace it with an empty matrix []. You can therefore remove the line "sFiles = [];". We can also rename the output variable "sAvgData", to be a bit more specific.
Edit low-pass filter call: Change the input to sAvgData and the output to sAvgDataLow, this way you will be able to keep track of the two files if you need to use them independently later in the script.
Edit the call to the snapshot process: Change the input to sAvgDataLow, and remove the output parameter (we are not expecting output from it).
bst_process('CallProcess', 'process_snapshot', sAvgDataLow, [], ...
'target', 5, ... % Recordings time series
'modality', 1); % MEG (All)
Replace the last lines with the following, to export the report instead of opening in the report viewer (edit the file path to point at your own user folder instead).
Select the code for the first process in the Matlab editor, right-click > Evaluate selection (or press F9).
If you haven't executed your script yet, you will get the following error in the Matlab command window:
Undefined variable "SubjectNames" or class "SubjectNames".
The variable SubjectNames is not defined yet: Execute the first two lines "SubjectNames = {'Subject01'}", then try again. You should now have a new variable in your Matlab workspace, which points at the two average files. Type "sAvgData(1)" in your command window to display the first element:
The field "sAvgData(1).FileName" contains the relative path to the to the Deviant average in the first run. This structure sAvgData contains also a lot of information that can be helpful in your script:
iStudy / iItem: Reference of the file in the database (described later in this tutorial).
FileType: 'raw' (continuous files), 'data' (recordings), 'results' (sources), 'timefreq' (time-frequency, spectrum and connectivity), or 'matrix' (any time series extracted from other files).
Comment: Comment field of the file (what is displayed in the database explorer)
Condition: Name of the condition/folder in which the file is located
SubjectFile: Relative path to the subject file (brainstormsubject.mat)
SubjectName: Name of the subject
DataFile: For types 'results' or 'timefreq', path of the parent file in the database explorer
ChannelFile: Relative path to the channel file
ChannelTypes: Cell array of channel types available for the input file
Naming conventions
To help you navigate in the Brainstorm code, here are some naming conventions:
Structures: Name starting with a "s" followed by a capital letter (eg. sFiles, sStudy, sSubject).
Indices: Either loop variables or array indices, name starting with a "i" (eg. iSubject, iStudy, iTime).
Counts: Number of elements in a group, name starting with a "n" (eg. nAvg, nTrials, nSubjects).
File names: Scripts and functions, only lower case, separation with "_" (eg. process_fft, bst_get).
Sub-functions: Inside a .m file, name starting with a capital, CamelCase (eg. CallProcess, Start).
Graphic handles: Matlab graphic objects, name starting with a "h" (eg. hFig, hAxes, hLine, hText).
Running the script
The simplified script looks like this:
% Input files
SubjectNames = {'Subject01'};
% Start a new report
bst_report('Start');
% Process: Select data files in: Subject01/*/Avg: deviant
sAvgData = bst_process('CallProcess', 'process_select_files_data', [], [], ...
'subjectname', SubjectNames{1}, ...
'tag', 'Avg: deviant');
% Process: Low-pass:30Hz
sAvgDataLow = bst_process('CallProcess', 'process_bandpass', sAvgData, [], ...
'highpass', 0, ...
'lowpass', 30, ...
'sensortypes', 'MEG');
% Process: Snapshot: Recordings time series
bst_process('CallProcess', 'process_snapshot', sAvgDataLow, [], ...
'target', 5, ... % Recordings time series
'modality', 1); % MEG (All)
% Save and display report
ReportFile = bst_report('Save');
bst_report('Export', ReportFile, 'C:\Users\franc\Documents\report_test.html');
You have three ways to execute it:
Select all the lines (Ctrl+A) and evaluate it in Matlab (F9).
In the Editor toolbar of the Matlab environment, click on the button [Run].
Save the file, go to this folder with Matlab (or add it to your path) and type the name of the script in the command window (without the ".m" at the end).
At the end of the execution, nothing happens, because we indicated we wanted to just export the report instead of opening it. To check out the report of execution: use the menu File > Report viewer from the Brainstorm window, or open the report_test.html that was saved somewhere on your computer.
On this page, you can get review everything that happened in the script: when it was executed, how long it took, what are the processes that were executed, some produced messages (two files were selected with the first process), and at the end you have the screen captures taken by process_snapshot.
Running the script again
If you execute the script again, it will not behave as expected anymore. The selection process we used assumes that there is only one file per folder with a comment that includes "Avg: deviant". This is not the case anymore after the execution, because the low-pass filtered files also contain the same tags. The execution of the first process of the script now returns 4 files.
In order to exclude the low-pass filtered files from this selection, you can add another process that will refine the selection. Use the script generator again to create a template call for another process, then copy-paste it in your script.
In Process1: Select any recordings file (we will not run anything, just generate a script).
Select process File > Select files: By tag: Search="low", Search the comment, Ignore the files.
Select the menu Generate .m script (make sure you do not overwrite the script you are currently working on), then close the pipeline editor.
Copy-paste and edit the call to process_select_tag to your main script.
Now the file selection part of your script should look like this, and should return only two files:
% Process: Select data files in: Subject01/*/Avg: Deviant
sAvgData = bst_process('CallProcess', 'process_select_files_data', [], [], ...
'subjectname', SubjectNames{1}, ...
'tag', 'Avg: deviant');
% Process: Ignore file comments with tag: low
sAvgData = bst_process('CallProcess', 'process_select_tag', sAvgData, [], ...
'tag', 'low', ...
'search', 2, ... % Search the file comments
'select', 2); % Ignore the files with the tag
With this last modification, your script is more robust. It can be executed multiple times without completely changing its behavior. When you are fetching files from the database using comment tags or file names, always pay attention to this aspect: the database grows and the further you go, the more specific your requests have to be.
A good practice can be to tag explicitely the output files your script generates if you need to fetch them later. You can use the process File > Add tag and File > Set comment.
Starting Brainstorm
Brainstorm must be running in the background for these scripts to run properly. The interface doesn't have to be visible on the screen, but the database engine must be running for processing requests. At the beginning of your script, you can explicitely start or restart Brainstorm.
brainstorm: Start Brainstorm with the regular GUI.
brainstorm nogui: Start in silent mode, without the GUI, without progress bar, without user interactions.
If you want to start Brainstorm only if Brainstorm is not already running, you can use the following code:
if ~brainstorm('status')
brainstorm nogui
end
To select a specific protocol at the beginning of your script:
ProtocolName = 'TutorialIntroduction';
% Get the protocol index
iProtocol = bst_get('Protocol', ProtocolName);
if isempty(iProtocol)
error(['Unknown protocol: ' ProtocolName]);
end
% Select the current procotol
gui_brainstorm('SetCurrentProtocol', iProtocol);
The functions bst_get and bst_set allow you to query the database, access the configuration of the software and some display parameters. The complete reference documentation of these functions is included directly in their code (brainstorm3/toolbox/core/bst_get.m and bst_set.m).
Let's start with a few simple examples:
>> ProtocolInfo = bst_get('ProtocolInfo') % Returns the configuration of the current protocol
ProtocolInfo =
Comment: 'TutorialIntroduction'
STUDIES: 'C:\Work\Protocols\TutorialIntroduction\data'
SUBJECTS: 'C:\Work\Protocols\TutorialIntroduction\anat'
iStudy: 6
UseDefaultAnat: 0
UseDefaultChannel: 0
>> isGUI = bst_get('isGUI') % Is the Brainstorm interface currently displayed (0=no, 1=yes)
>> bst_set('FlipYAxis', 1) % The new figures will have the Y axis flipped
>> bst_set('TSDisplayMode', 'butterfly') % The new figures will use a "butterfly" view
>> bst_set('FieldTripDir', FieldTripDir) % Set the path where the FieldTrip toolbox is installed
To reference the files in the database, each protocol is subdivided in Subjects (the "anat" folder and containing the MRI, surfaces and atlases) and Studies (the "data" folder, including the recordings, channel files and all the analyses). Each Study corresponds to a sub-folder (eg. protocol/data/subject01/run01/), and is attached to only one subject.
Subjects and Studies are referenced in the protocol with a unique index most of the time kept in variables named iSubject and iStudy. The files available in them are also referenced with indices, with variables such as iAnatomy, iSurface, iData, iHeadModel, iResults, iTimefreq. You can see the indices in the database explorer by hovering your mouse over the nodes of the tree:
Example: Getting the study structure from the variable sAvgData, defined in the script:
>> sAvgData(1)
ans =
iStudy: 6
iItem: 1
...
>> sStudy = bst_get('Study', sAvgData(1).iStudy) % Get a study structure with its index
sStudy =
Name: 'S01_AEF_20131218_01_600Hz_notch'
FileName: 'Subject01/S01_AEF_20131218_01_600Hz_notch/brainstormstudy.mat'
DateOfStudy: '13-May-2016'
BrainStormSubject: 'Subject01/brainstormsubject.mat' % Corresponding subject filename
Condition: {'S01_AEF_20131218_01_600Hz_notch'} % Name of the condition (=folder)
Channel: [1x1 struct] % Channel file for the folder
iChannel: [] % Not used anymore
Data: [1x242 struct] % List of "data" files in the folder
HeadModel: [1x1 struct] % List of head models in the folder
iHeadModel: 1 % Default head model (file in green)
Result: [1x244 struct] % List of source files and source links
Stat: [1x0 struct] % List of statistical results
Image: [0x0 struct] % List of images
NoiseCov: [1x2 struct] % Noise(1) and data(2) covariance files
Dipoles: [0x0 struct] % List of dipole files in the folder
Timefreq: [1x247 struct] % List of time-frequency files
Matrix: [0x0 struct] % List of "matrix" files in the folder
% If 1, the trial is marked as bad in the explorer
Example: Getting the data structure.
>> sData = sStudy.Data(sAvgData(1).iItem) % Get the structure representing the file from sStudy
sData =
FileName: 'Subject01/S01_AEF_20131218_01_600Hz_notch/data_deviant_average_160513_1329.mat'
Comment: 'Avg: deviant (39 files)' % File comment, displayed in the database explorer
DataType: 'recordings' % Type of data in the file (mostly "recordings")
BadTrial: 0
Example: Getting the subject structure.
>> sSubject = bst_get('Subject', sStudy.BrainStormSubject) % Get subject structure from filename
sSubject =
Name: 'Subject01' % Subject name, same as folder name
Comments: '' % Not used much
FileName: 'Subject01/brainstormsubject.mat'
DateOfAcquisition: '' % Not used anymore
Anatomy: [1x1 struct] % List of MRI volumes in the subject folder
Surface: [1x9 struct] % List of surfaces in the subject folder
iAnatomy: 1 % Default MRI volume (in green, index of array Anatomy)
iScalp: 9 % Default head surface (index of array Surface)
iCortex: 4 % Default cortex surface (index of array Surface)
iInnerSkull: [] % Default inner skull surface (index of array Surface)
iOuterSkull: [] % Default outer skull surface (index of array Surface)
iOther: [] % Not used anymore
UseDefaultAnat: 0 % If 1: Subject uses the default anatomy
UseDefaultChannel: 0 % 0=one channel/folder, 1=one/subject, 2=one global
Example: Getting the study structure and data index from a file name.
Many other options are available for searching files in the database with bst_get. We cannot list them all in this page, but you can refer to the code of bst_get.m for more information.
1 function [argout1, argout2, argout3, argout4, argout5] = bst_get( varargin )
2 % BST_GET: Get a Brainstorm structure.
3 % This function is used to abstract the way that these structures are stored.
4 %
5 % USAGE :
6 % ====== DIRECTORIES ==================================================================
7 % - bst_get('UserDir') : User home directory
8 % - bst_get('BrainstormHomeDir') : Application directory of brainstorm
9 % - bst_get('BrainstormUserDir') : User home directory for brainstorm (<home>/.brainstorm/)
10 % - bst_get('BrainstormTmpDir') : User brainstorm temporary directory (Default: <home>/.brainstorm/tmp/)
11 % - bst_get('BrainstormTmpDir', isForcedDefault=0, SubDir=[]) : User DEFAULT brainstorm temporary directory (<home>/.brainstorm/tmp/SubDir)
12 % - bst_get('BrainstormDocDir') : Doc folder folder of the Brainstorm distribution (may vary in compiled versions)
13 % - bst_get('BrainstormDefaultsDir') : Defaults folder of the Brainstorm distribution (may vary in compiled versions)
14 % - bst_get('UserReportsDir') : User reports directory (<home>/.brainstorm/reports/)
15 % - bst_get('UserMexDir') : User temporary directory (<home>/.brainstorm/mex/)
16 % - bst_get('UserProcessDir') : User custom processes directory (<home>/.brainstorm/process/)
17 % - bst_get('UserDefaultsDir') : User defaults directory (<home>/.brainstorm/defaults/)
18 % - bst_get('UserPluginsDir') : User plugins directory (brainstorm3/plugins/ if it exists, otherwise <home>/.brainstorm/plugins/)
19 % - bst_get('BrainstormDbFile') : User brainstorm.mat file (<home>/.brainstorm/brainstorm.mat)
20 % - bst_get('BrainstormDbDir') : User database directory (contains all the brainstorm protocols)
21 % - bst_get('DirDefaultSubject') : Directory name of the default subject
22 % - bst_get('DirDefaultStudy') : Directory name of the default study for each subject
23 % - bst_get('DirAnalysisInter') : Directory name of the inter-subject analysis study
24 % - bst_get('DirAnalysisIntra') : Directory name of the intra-subject analysis study (for each subject)
25 % - bst_get('AnatomyDefaults') : Get the contents of directory bstDir/defaults/anatomy
26 % - bst_get('MniAtlasDefaults') : Get the contents of directory bstDir/defaults/mniatlas
27 % - bst_get('EegDefaults') : Get the contents of directory bstDir/defaults/eeg
28 % - bst_get('LastUsedDirs') : Structure with all the last used directories (last used)
29 % - bst_get('OsType', isMatlab=1) : Get a string that describes the operating system (if isMatlab=1 return the Matlab/JVM platform, else return the real host system)
30 % - bst_get('FileFilters', DataType) : Get the list of import filters for a specific data type
31 % - bst_get('PluginCustomPath') : Full custom path to all plugins
32 % - bst_get('BrainSuiteDir') : Full path to a local installation of BrainSuite
33 % - bst_get('SpmTpmAtlas') : Full path to the SPM atlas TPM.nii
34 % - bst_get('PythonExe') : Path to the python executable
35 %
36 % ====== PROTOCOLS ====================================================================
37 % - bst_get('iProtocol') : Indice of current protocol
38 % - bst_get('Protocol', ProtocolName): Return the indice of the protocol ProtocolName, or [] if it doesn't exist
39 % - bst_get('ProtocolInfo') : Definition structure for current protocol
40 % - bst_get('ProtocolSubjects') : Subjects list for current protocol
41 % - bst_get('ProtocolStudies') : Studies list for current protocol
42 % - bst_get('isProtocolLoaded') : 1 if the protocol has been loaded, 0 else
43 % - bst_get('isProtocolModified') : 1 if the protocol has been modified, 0 else
44 %
45 % ====== STUDIES ======================================================================
46 % - bst_get('Study', StudyFileName) : Get one study in current protocol with its file name
47 % - bst_get('Study', iStudies) : Get one or more studies
48 % - bst_get('Study') : Get current study in current protocol
49 % - bst_get('StudyCount') : Get number of studies in the current protocol
50 % - bst_get('StudyWithSubject', SubjectFile) : Find studies associated with a given subject file (WITHOUT the system studies ('intra_subject', 'default_study'))
51 % - bst_get('StudyWithSubject', ..., 'intra_subject') : Find studies ... INCLUDING 'intra_subject' study
52 % - bst_get('StudyWithSubject', ..., 'default_study') : Find studies ... INCLUDING 'default_study' study
53 % - bst_get('StudyWithCondition', ConditionPath) : Find studies for a given condition path
54 % - bst_get('ChannelStudiesWithSubject', iSubjects) : Get all the studies where there should be a channel file for a list of subjects
55 % - bst_get('AnalysisIntraStudy', iSubject) : Get the default analysis study for target subject
56 % - bst_get('AnalysisInterStudy') : Get the default analysis study for inter-subject analysis
57 % - bst_get('DefaultStudy', iSubject) : Get the default study for target subject (by subject indice)
58 % - bst_get('DefaultStudy') : Get the global default study (common to all subjects)
59 % - bst_get('DefaultStudy', SubjectFile) : Get the default study for target subject (by filename)
60 % - bst_get('ChannelFile', ChannelFile) : Find a channel file in current protocol
61 % - bst_get('ChannelFileForStudy', StudyFile/DataFile) : Find a channel file in current protocol
62 % - bst_get('ChannelForStudy', iStudies) : Return current Channel struct for target study
63 % - bst_get('ChannelModalities', ChannelFile) : Return displayable modalities for a Channel file
64 % - bst_get('ChannelModalities', DataFile) : Return displayable modalities for a Data/Results/Timefreq... file
65 % - bst_get('ChannelDevice', ChannelFile) : Return acquisistion device for a channel file
66 % - bst_get('TimefreqDisplayModalities', TfFile) : Get displayable modalities for a TF file based on recordings
67 % - bst_get('HeadModelForStudy', iStudy) : Return current HeadModel struct for target study
68 % - bst_get('HeadModelFile', HeadModelFile) : Find a HeadModel file in current protocol
69 % - bst_get('HeadModelFile', HeadModelFile, iStudies): Find a HeadModel file in current protocol
70 % - bst_get('NoiseCovFile', NoiseCovFile) : Find a noise covariance file file in current protocol
71 % - bst_get('NoiseCovFile', NoiseCovFile, iStudies) : Find a noise covariance file in current protocol
72 % - bst_get('NoiseCovFile', DataCovFile) : Find a data covariance file file in current protocol
73 % - bst_get('NoiseCovFile', DataCovFile, iStudies) : Find a data covariance file file in current protocol
74 % - bst_get('DataFile', DataFile) : Find a DataFile in current protocol
75 % - bst_get('DataFile', DataFile, iStudies) : Find a DataFile in current protocol
76 % - bst_get('DataForDataList', iStudy, DataListName) : Find all the DataFiles grouped by a data list
77 % - bst_get('DataForStudy', iStudy) : Find all the Data files that are dependent on the channel/headmodel of a given study
78 % - bst_get('DataForStudies', iStudies)
79 % - bst_get('DataForChannelFile', ChannelFile) : Find all the DataFiles that use the given ChannelFile
80 % - bst_get('ResultsFile', ResultsFile) : Find a ResultsFile in current protocol
81 % - bst_get('ResultsFile', ResultsFile, iStudies) : Find a ResultsFile in input studies
82 % - bst_get('ResultsForDataFile', DataFile, iStudies): Find all results computed based on DataFile
83 % - bst_get('StatFile', StatFile) : Find a StatFile in current protocol
84 % - bst_get('StatFile', StatFile, iStudies) : Find a StatFile in input studies
85 % - bst_get('StatForDataFile', DataFile, iStudies)
86 % - bst_get('StatForDataFile', DataFile)
87 % - bst_get('TimefreqFile', TimefreqFile)
88 % - bst_get('TimefreqFile', TimefreqFile, iStudies)
89 % - bst_get('TimefreqForFile', FileName, iStudies) : Find all timefreq files computed based on target file
90 % - bst_get('TimefreqForKernel', KernelFile)
91 % - bst_get('DipolesFile', DipolesFile)
92 % - bst_get('DipolesFile', DipolesFile, iStudies)
93 % - bst_get('DipolesForFile', FileName, iStudies) : Find all dipoles files computed based on target file
94 % - bst_get('DipolesForKernel', KernelFile)
95 % - bst_get('MatrixFile', MatrixFile)
96 % - bst_get('MatrixFile', MatrixFile, iStudies)
97 % - bst_get('MatrixForMatrixList', iStudy, MatrixListName)
98 % - bst_get('AnyFile', AnyFile)
99 % - bst_get('AnyFile', AnyFile, iStudies)
100 % - bst_get('RelatedDataFile', FileName)
101 % - bst_get('RelatedDataFile', FileName, iStudies)
102 % - bst_get('GetFilenames')
103 %
104 % ====== SUBJECTS ======================================================================
105 % - bst_get('Subject', SubjectFileName, isRaw) : Find a subject in current protocol with its file name
106 % - bst_get('Subject', SubjectName, isRaw) : Find a subject in current protocol with its name
107 % - bst_get('Subject', iSubject) : Get a subject (normal or default if iSubject==0)
108 % - bst_get('Subject') : Get current subject in current protocol
109 % - bst_get('SubjectCount') : Get number of studies in the current protocol
110 % - bst_get('NormalizedSubjectName') : Name of the subject with a normalized anatomy
111 % - bst_get('NormalizedSubject') : Get group analysis subject for the current protocol
112 % - bst_get('ConditionsForSubject', SubjectFile) : Find all conditions for a given subject
113 % - bst_get('SurfaceFile', SurfaceFile) : Find a surface in current protocol
114 % - bst_get('SurfaceFileByType', iSubject, SurfaceType) : Find surfaces with given type for subject #i (default only)
115 % - bst_get('SurfaceFileByType', SubjectFile, SurfaceType) : Find surfaces with given type for subject SubjectFile (default only)
116 % - bst_get('SurfaceFileByType', SurfaceName, SurfaceType) : Find surfaces with given type for subject that also has surface SurfaceName (default only)
117 % - bst_get('SurfaceFileByType', MriName, SurfaceType) : Find surfaces with given type for subject that also has MRI MriName (default only)
118 % - bst_get('SurfaceFileByType', ..., ..., isDefaultOnly) : If 0, return all the surfaces of the given type, instead of only the default surface
119 % - bst_get('MriFile', MriFile) : Find a MRI in current protocol
120 %
121 % ====== GUI =================================================================
122 % - bst_get('BstControls') : Return main Brainstorm GUI structure
123 % - bst_get('BstFrame') : Return main Brainstorm JFrame
124 % - bst_get('isGUI') : Return 1 if the Brainstorm interface is displayed
125 % - bst_get('GuiLevel') : Return GUI level: -1=server, 0=nogui, 1=normal, 2=autopilot
126 % - bst_get('ScreenDef') : Get screens configuration
127 % - bst_get('DecorationSize') : Get dimensions of the windows decorations
128 % - bst_get('Layout') : Configuration of the main Brainstorm window
129 % - bst_get('Layout', prop) : Get one property in the layout properties
130 % - bst_get('PanelContainer') : Display list of registered panel containers
131 % - bst_get('PanelContainer', ContainerName) : Get a panel container handle
132 % - bst_get('Panel') : Display list of registered panels
133 % - bst_get('Panel', PanelName) : Find a panel with its name
134 % - bst_get('PanelControls', PanelName) : Get the controls of a panel
135 % - bst_get('Clipboard') : Nodes that were copied from the interface
136 % - bst_get('FigFont') : Standard font size displayed in the figures
137 % - bst_get('Font') : Create a Java font, scaled for the operating system
138 %
139 % ====== CONFIGURATION =================================================================
140 % - bst_get('Version') : Brainstorm version
141 % - bst_get('MatlabVersion') : Matlab version (version number * 100, eg. 801)
142 % - bst_get('MatlabReleaseName') : Matlab version (release name, eg. "R2014a")
143 % - bst_get('JavaVersion') : Java version
144 % - bst_get('isJavacomponent') : Returns 1 if javacomponent is available (Matlab < 2019b), 0 otherwise
145 % - bst_get('SystemMemory') : Amount of Total and Available physical memory (RAM), in MiB
146 % - bst_get('ByteOrder') : {'l','b'} - Byte order used to read and save binary files
147 % - bst_get('TSDisplayMode') : {'butterfly','column'}
148 % - bst_get('ElectrodeConfig', Modality) : Structure describing the display values for SEEG/ECOG/EEG contacts
149 % - bst_get('ElecInterpDist', Modality) : Structure describing the maximum distance for interpolating SEEG/ECOG/EEG values onto a surface
150 % - bst_get('AutoUpdates') : {0,1} - If 1, check automatically for updates at startup
151 % - bst_get('ForceMatCompression') : {0,1} - If 1, always save mat-files using the v7 format instead of v6
152 % - bst_get('IgnoreMemoryWarnings') : {0,1} - If 1, do not display memory warnings at the Brainstorm startup
153 % - bst_get('ExpertMode') : {0,1} - If 1, show advanced options that regular user do not see
154 % - bst_get('DisplayGFP') : {0,1} - If 1, the GFP is displayed on all the time series figures
155 % - bst_get('DownsampleTimeSeries') : {0,1,...} - If > 0, downsample dense time series for faster display
156 % - bst_get('DisableOpenGL') : {0,1,2} - If 1, do not use OpenGL renderer; if 2, use software OpenGL
157 % - bst_get('InterfaceScaling') : {100,125,150,...} - Scales the Brainstorm GUI by a fixed factor
158 % - bst_get('GraphicsSmoothing') : {0,1} - If 1, uses the graphics smoothing (Matlab >= 2014b)
159 % - bst_get('SystemCopy') : {0,1} - If 1, uses the system calls mv/cp instead of movefile/copyfile (Linux only)
160 % - bst_get('DefaultFormats') : Default formats for importing/exporting data, channels, ... (last used)
161 % - bst_get('BFSProperties') : Conductivities and thicknesses for 3-shell spherical forward model
162 % - bst_get('ImportDataOptions') : Import options for recordings
163 % - bst_get('ImportEegRawOptions') : Importation options for RAW EEG format
164 % - bst_get('BugReportOptions') : Bug reporter options
165 % - bst_get('DefaultSurfaceDisplay') : Default display options for surfaces (smooth, data threshold, sulci map)
166 % - bst_get('MagneticExtrapOptions') : Structure with the options for magnetic field extrapolation
167 % - bst_get('DefaultFreqBands')
168 % - bst_get('TimefreqOptions_morlet')
169 % - bst_get('TimefreqOptions_fft')
170 % - bst_get('TimefreqOptions_psd')
171 % - bst_get('TimefreqOptions_hilbert')
172 % - bst_get('OpenMEEGOptions')
173 % - bst_get('DuneuroOptions')
174 % - bst_get('GridOptions_headmodel')
175 % - bst_get('GridOptions_dipfit')
176 % - bst_get('UniformizeTimeSeriesScales') : {0,1} - If 1, the Y-axis of all the time series figures have the scale
177 % - bst_get('FlipYAxis') : {0,1} - If 1, the recordings are displayed with the negative values UP
178 % - bst_get('AutoScaleY') : {0,1} - If 1, the axis limits are updated when the figure is updated
179 % - bst_get('ShowXGrid') : {0,1} - If 1, show the XGrid in the time series figures
180 % - bst_get('ShowYGrid') : {0,1} - If 1, show the YGrid in the time series figures
181 % - bst_get('ShowZeroLines') : {0,1} - If 1, show the Y=0 lines in the columns view
182 % - bst_get('ShowEventsMode') : {'dot','line','none'}
183 % - bst_get('Resolution') : [resX,resY] fixed resolutions for X and Y axes
184 % - bst_get('FixedScaleY', Modality) : Struct with the scales to impose on the recordings for the selected modality
185 % - bst_get('XScale', XScale) : {'log', 'linear'}
186 % - bst_get('YScale', YScale) : {'log', 'linear'}
187 % - bst_get('UseSigProcToolbox') : Use Matlab's Signal Processing Toolbox when available
188 % - bst_get('RawViewerOptions', sFile) : Display options for RAW recordings, adapt for specific file
189 % - bst_get('RawViewerOptions') : Display options for RAW recordings
190 % - bst_get('TopoLayoutOptions') : Display options for 2DLayout display
191 % - bst_get('StatThreshOptions') : Options for online statistical thresholding
192 % - bst_get('ContactSheetOptions') : Display options for contact sheets
193 % - bst_get('ProcessOptions') : Options related with the data processing
194 % - bst_get('CustomColormaps') : Gets the list of user defined colormaps
195 % - bst_get('MriOptions') : Configuration for MRI display
196 % - bst_get('DigitizeOptions') : Digitizer options
197 % - bst_get('PcaOptions') : Options for PCA dimension reduction (xyz flattening or scout function)
198 % - bst_get('ReadOnly') : Read only interface
199 % - bst_get('NodelistOptions') : Structure with the options for file selection in the Process1 and Process2 panels
200 % - bst_get('ResizeFunction') : Get the appropriate resize function
201 % - bst_get('groot') : Get the root graphic object
202 % - bst_get('JFrame', hFig) : Get the underlying java frame for a Matlab figure
203 % - bst_get('LastPsdDisplayFunction') : Display option of measure for spectrum (log, power, magnitude, etc.)
204 % - bst_get('LastTfDisplayFunction') : Display option of measure for timefreq map (log, power, magnitude, etc.)
205 % - bst_get('PlotlyCredentials') : Get the credentials and URL to connect to plot.ly server
206 % - bst_get('ExportBidsOptions') : Additional metadata for BIDS export
207 % - bst_get('Pipelines') : Saved Pipelines stored
208 %
209 % SEE ALSO bst_set
210
211 % @=============================================================================
212 % This function is part of the Brainstorm software:
213 % https://neuroimage.usc.edu/brainstorm
214 %
215 % Copyright (c) University of Southern California & McGill University
216 % This software is distributed under the terms of the GNU General Public License
217 % as published by the Free Software Foundation. Further details on the GPLv3
218 % license can be found at http://www.gnu.org/copyleft/gpl.html.
219 %
220 % FOR RESEARCH PURPOSES ONLY. THE SOFTWARE IS PROVIDED "AS IS," AND THE
221 % UNIVERSITY OF SOUTHERN CALIFORNIA AND ITS COLLABORATORS DO NOT MAKE ANY
222 % WARRANTY, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO WARRANTIES OF
223 % MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE, NOR DO THEY ASSUME ANY
224 % LIABILITY OR RESPONSIBILITY FOR THE USE OF THIS SOFTWARE.
225 %
226 % For more information type "brainstorm license" at command prompt.
227 % =============================================================================@
228 %
229 % Authors: Francois Tadel, 2008-2023
230 % Martin Cousineau, 2017
231 % Raymundo Cassani, 2021-2023
232
233 %% ==== PARSE INPUTS ====
234 global GlobalData;
235 if ((nargin >= 1) && ischar(varargin{1}))
236 contextName = varargin{1};
237 else
238 return
239 end
240 % Initialize returned variable
241 argout1 = [];
242 argout2 = [];
243 argout3 = [];
244 argout4 = [];
245 argout5 = [];
246
247 % Get required context structure
248 switch contextName
249 %% ==== SUBFUNCTIONS =====
250 case 'findFileInStudies'
251 [argout1, argout2, argout3] = findFileInStudies(varargin{2:end});
252
253 %% ==== BRAINSTORM CONFIGURATION ====
254 case 'Version'
255 argout1 = GlobalData.Program.Version;
256
257 case 'MatlabVersion'
258 if ~exist('version','builtin')
259 Version = 601;
260 else
261 % Get version number
262 str_vers = version();
263 vers = sscanf(str_vers, '%d.%d%*s');
264 if isempty(vers) || any(~isnumeric(vers)) || any(isnan(vers))
265 vers = 0;
266 end
267 Version = 100 * vers(1) + vers(2);
268 % Get release name
269 ipar = [find(str_vers == '('), find(str_vers == ')')];
270 end
271 argout1 = Version;
272
273 case 'MatlabReleaseName'
274 if ~exist('version','builtin')
275 Release = 'Matlab6';
276 else
277 % Get version number
278 str_vers = version();
279 % Get release name
280 ipar = [find(str_vers == '('), find(str_vers == ')')];
281 Release = str_vers(ipar(1)+1:ipar(2)-1);
282 end
283 argout1 = Release;
284
285 case 'JavaVersion'
286 strver = char(java.lang.System.getProperty('java.version'));
287 iDot = find(strver == '.');
288 if (length(iDot) < 2)
289 argout1 = str2num(strver);
290 else
291 argout1 = str2num(strver(1:iDot(2)-1));
292 end
293
294 case 'isJavacomponent'
295 % After Matlab 2019b, javacomponent() and JavaFrame property have been deprecated
296 argout1 = (bst_get('MatlabVersion') <= 906);
297
298 case 'SystemMemory'
299 % [RamTotal_MiB, RamAvailable_MiB] = bst_get('SystemMemory')
300 RamTotal_MiB = [];
301 RamAvailable_MiB = [];
302 tmp = regexp(bst_get('OsType'), '^[a-z]+', 'match', 'ignorecase');
303 if ~isempty(tmp)
304 osFamily = tmp{1};
305 end
306 if strcmpi(osFamily, 'win') && (bst_get('MatlabVersion') >= 706)
307 try
308 % Get memory info
309 [usermem, systemmem] = memory();
310 RamTotal_MiB = round(systemmem.PhysicalMemory.Total / 1024 / 1024);
311 RamAvailable_MiB = round(usermem.MemAvailableAllArrays / 1024 / 1024);
312 catch
313 % Whatever...
314 end
315
316 elseif strcmpi(osFamily, 'linux')
317 try
318 meminfoRes = fileread('/proc/meminfo');
319 ramTotalkB = regexp(meminfoRes, '(?<=MemTotal:)(.*?)(?=kB)', 'match');
320 ramAvailablekB = regexp(meminfoRes, '(?<=MemAvailable:)(.*?)(?=kB)', 'match');
321 if ~isempty(ramAvailablekB) && ~isempty(ramTotalkB)
322 ramTotalkB = str2double(strtrim(ramTotalkB{1}));
323 RamTotal_MiB = round(ramTotalkB /1024);
324 ramAvailablekB = str2double(strtrim(ramAvailablekB{1}));
325 RamAvailable_MiB = round(ramAvailablekB /1024);
326 end
327 catch
328 % Whatever...
329 end
330 elseif strcmpi(osFamily, 'mac')
331 try
332 [~, mem_pressure] = system('memory_pressure');
333 if ~isempty(mem_pressure)
334 ramTotalB = regexp(mem_pressure, '(?<=The system has)(.*?)(?= )', 'match');
335 prcFree = regexp(mem_pressure, '(?<=System-wide memory free percentage:)(.*?)(?=%)', 'match');
336 if ~isempty(ramTotalB) && ~isempty(prcFree)
337 ramTotalB = str2double(ramTotalB{1});
338 RamTotal_MiB = round(ramTotalB / 1024 / 1024);
339 ramAvailableB = ramTotalB * str2double(prcFree{1}) / 100;
340 RamAvailable_MiB = round(ramAvailableB / 1024 / 1024);
341 end
342 end
343 catch
344 % Whatever...
345 end
346 end
347 argout1 = RamTotal_MiB;
348 argout2 = RamAvailable_MiB;
349
350 case 'BrainstormHomeDir'
351 argout1 = GlobalData.Program.BrainstormHomeDir;
352 case 'BrainstormDbDir'
353 argout1 = GlobalData.DataBase.BrainstormDbDir;
354 case 'UserDir'
355 try
356 if ispc
357 userDir = getenv('USERPROFILE');
358 else
359 userDir = char(java.lang.System.getProperty('user.home'));
360 end
361 catch
362 userDir = '';
363 end
364 if isempty(userDir)
365 userDir = bst_get('BrainstormHomeDir');
366 end
367 argout1 = userDir;
368
369 case 'BrainstormUserDir'
370 bstUserDir = bst_fullfile(bst_get('UserDir'), '.brainstorm');
371 if ~isdir(bstUserDir)
372 res = mkdir(bstUserDir);
373 if ~res
374 error(['Cannot create Brainstorm user directory: "' bstUserDir '".']);
375 end
376 end
377 argout1 = bstUserDir;
378
379 case 'BrainstormTmpDir'
380 tmpDir = '';
381 isForcedDefault = ((nargin >= 2) && ~isempty(varargin{2}) && varargin{2});
382 % Subdirectory
383 if (nargin >= 3) && ~isempty(varargin{3})
384 SubDir = varargin{3};
385
386 % % Add PID of MATLAB process to the temporary folder
387 % try
388 % pid = feature('getpid');
389 % catch
390 % pid = [];
391 % end
392 % if ~isempty(pid)
393 % SubDir = sprintf('%s_%d', SubDir, feature('getpid'));
394 % end
395
396 % Add date+time of MATLAB process to the temporary folder
397 c = clock;
398 SubDir = sprintf('%s_%02.0f%02.0f%02.0f_%02.0f%02.0f%02.0f', SubDir, c(1)-2000, c(2:6));
399 else
400 SubDir = [];
401 end
402 % Default folder: userdir/tmp
403 defDir = bst_fullfile(bst_get('BrainstormUserDir'), 'tmp');
404 % If temporary directory is set in the preferences
405 if ~isForcedDefault && isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'BrainstormTmpDir')
406 tmpDir = GlobalData.Preferences.BrainstormTmpDir;
407 end
408 % Else: use directory userdir/tmp
409 if isempty(tmpDir)
410 tmpDir = defDir;
411 end
412 % Create directory if it does not exist yet
413 if ~isdir(tmpDir)
414 res = mkdir(tmpDir);
415 if ~res && ~strcmp(tmpDir, defDir)
416 disp(['BST> Cannot create Brainstorm temporary directory: "' tmpDir '".']);
417 disp(['BST> Using default directory instead: "' defDir '".']);
418 % Create temporary folder
419 tmpDir = defDir;
420 if ~isdir(tmpDir)
421 res = mkdir(tmpDir);
422 else
423 res = 1;
424 end
425 end
426 % Error: cannot create any temporary folder
427 if ~res
428 error(['Cannot create Brainstorm temporary directory: "' tmpDir '".']);
429 end
430 end
431 % Add sub-directory
432 if ~isempty(SubDir)
433 tmpDir = bst_fullfile(tmpDir, SubDir);
434 % Create directory if it does not exist yet
435 if ~isdir(tmpDir)
436 res = mkdir(tmpDir);
437 if ~res
438 error(['Cannot create Brainstorm temporary directory: "' tmpDir '".']);
439 end
440 end
441 end
442 argout1 = tmpDir;
443
444 case 'BrainstormDocDir'
445 docDir = bst_fullfile(GlobalData.Program.BrainstormHomeDir, 'doc');
446 if ~exist(docDir, 'file')
447 % Matlab compiler >= 2018b stores 'doc' under 'bst_javabuil'
448 docDir = bst_fullfile(GlobalData.Program.BrainstormHomeDir, 'bst_javabuil', 'doc');
449 if ~exist(docDir, 'file')
450 docDir = '';
451 disp('BST> Could not find "doc" folder.');
452 disp(['BST> BrainstormHomeDir = ' GlobalData.Program.BrainstormHomeDir]);
453 end
454 end
455 argout1 = docDir;
456
457 case 'BrainstormDefaultsDir'
458 defaultsDir = bst_fullfile(GlobalData.Program.BrainstormHomeDir, 'defaults');
459 if ~exist(defaultsDir, 'file')
460 % Matlab compiler >= 2018b stores 'defaults' under 'bst_javabuil'
461 defaultsDir = bst_fullfile(GlobalData.Program.BrainstormHomeDir, 'bst_javabuil', 'defaults');
462 if ~exist(defaultsDir, 'file')
463 defaultsDir = '';
464 disp('BST> Could not find "defaults" folder.');
465 disp(['BST> BrainstormHomeDir = ' GlobalData.Program.BrainstormHomeDir]);
466 end
467 end
468 argout1 = defaultsDir;
469
470 case 'UserReportsDir'
471 reportDir = bst_fullfile(bst_get('BrainstormUserDir'), 'reports');
472 if ~isdir(reportDir)
473 res = mkdir(reportDir);
474 if ~res
475 error(['Cannot create user reports directory: "' reportDir '".']);
476 end
477 end
478 argout1 = reportDir;
479
480 case 'UserMexDir'
481 mexDir = bst_fullfile(bst_get('BrainstormUserDir'), 'mex');
482 if ~isdir(mexDir)
483 res = mkdir(mexDir);
484 if ~res
485 error(['Cannot create Brainstorm mex-files directory: "' mexDir '".']);
486 end
487 end
488 argout1 = mexDir;
489
490 case 'UserProcessDir'
491 processDir = bst_fullfile(bst_get('BrainstormUserDir'), 'process');
492 if ~isdir(processDir)
493 res = mkdir(processDir);
494 if ~res
495 error(['Cannot create Brainstorm custom processes directory: "' processDir '".']);
496 end
497 end
498 argout1 = processDir;
499
500 case 'UserDefaultsDir'
501 defDir = bst_fullfile(bst_get('BrainstormUserDir'), 'defaults');
502 defDirAnat = bst_fullfile(defDir, 'anatomy');
503 defDirEeg = bst_fullfile(defDir, 'eeg');
504 if ~isdir(defDir)
505 res = mkdir(defDir);
506 if ~res
507 error(['Cannot create user templates directory: "' defDir '".']);
508 end
509 end
510 if ~isdir(defDirAnat)
511 res = mkdir(defDirAnat);
512 if ~res
513 error(['Cannot create user templates directory: "' defDirAnat '".']);
514 end
515 end
516 if ~isdir(defDirEeg)
517 res = mkdir(defDirEeg);
518 if ~res
519 error(['Cannot create user templates directory: "' defDirEeg '".']);
520 end
521 end
522 argout1 = defDir;
523
524 case 'UserPluginsDir'
525 % If brainstorm/plugins exists, use it (case of an environment exported and reused)
526 pluginsDir = bst_fullfile(GlobalData.Program.BrainstormHomeDir, 'plugins');
527 % Otherwise (standard case): <home>/.brainstorm/plugins/
528 if ~exist(pluginsDir, 'file')
529 pluginsDir = bst_fullfile(bst_get('BrainstormUserDir'), 'plugins');
530 if ~isdir(pluginsDir)
531 res = mkdir(pluginsDir);
532 if ~res
533 error(['Cannot create plugins directory: "' pluginsDir '".']);
534 end
535 end
536 end
537 argout1 = pluginsDir;
538
539 case 'BrainstormDbFile'
540 argout1 = bst_fullfile(bst_get('BrainstormUserDir'), 'brainstorm.mat');
541
542 case 'Pipelines'
543 argout1 = GlobalData.Processes.Pipelines;
544
545 case 'OsType'
546 switch (mexext)
547 case 'mexglx', argout1 = 'linux32';
548 case 'mexa64', argout1 = 'linux64';
549 case 'mexmaci', argout1 = 'mac32';
550 case 'mexmaci64', argout1 = 'mac64';
551 case 'mexmaca64', argout1 = 'mac64arm';
552 case 'mexs64', argout1 = 'sol64';
553 case 'mexw32', argout1 = 'win32';
554 case 'mexw64', argout1 = 'win64';
555 otherwise, error('Unsupported extension.');
556 end
557 % CALL: bst_get('OsType', isMatlab=0)
558 if (nargin >= 2) && isequal(varargin{2}, 0)
559 if strcmpi(argout1, 'win32') && (~isempty(strfind(java.lang.System.getProperty('java.home'), '(x86)')) || ~isempty(strfind(java.lang.System.getenv('ProgramFiles(x86)'), '(x86)')))
560 argout1 = 'win64';
561 end
562 end
563
564 case 'OsName'
565 argout1 = '';
566 osFamily = [];
567 tmp = regexp(bst_get('OsType'), '^[a-z]+', 'match', 'ignorecase');
568 if ~isempty(tmp)
569 osFamily = tmp{1};
570 end
571 switch osFamily
572 case 'win'
573 [~, system_info] = system('ver');
574 argout1 = strtrim(system_info);
575
576 case 'linux'
577 os_release = fileread('/etc/os-release');
578 osName = regexp(os_release, '(?<=PRETTY_NAME=")(.*?)(?=")', 'match');
579 if ~isempty(osName)
580 osName = strtrim(osName{1});
581 else
582 osName = regexp(os_release, '(?<=NAME=")(.*?)(?=")', 'match');
583 if ~isempty(osName)
584 osName = strtrim(osName{1});
585 else
586 osName = 'Linux unknow distribution';
587 end
588 end
589 [~, kernelVer] = system('uname -r');
590 kernelVer = strtrim(kernelVer);
591 argout1 = [osName, ' (' kernelVer, ')'];
592
593 case 'mac'
594 [~, sw_vers] = system('sw_vers');
595 osName = regexp(sw_vers, '(?<=ProductName:)(.*?)(?=\n)', 'match');
596 osName = strtrim(osName{1});
597 osVer = regexp(sw_vers, '(?<=ProductVersion:)(.*?)(?=\n)', 'match');
598 osVer = strtrim(osVer{1});
599 [~, osHw] = system('uname -m');
600 osHw = strtrim(osHw);
601 argout1 = [osName, ' ' osVer, ' (', osHw, ')'];
602 end
603
604
605 %% ==== PROTOCOL ====
606 case 'iProtocol'
607 if isempty(GlobalData.DataBase.iProtocol)
608 argout1 = 0;
609 else
610 argout1 = GlobalData.DataBase.iProtocol;
611 end
612
613 case 'Protocol'
614 if ~isempty(GlobalData.DataBase.ProtocolInfo)
615 argout1 = find(strcmpi({GlobalData.DataBase.ProtocolInfo.Comment}, varargin{2}));
616 else
617 argout1 = [];
618 end
619
620 case {'ProtocolInfo', 'ProtocolSubjects', 'ProtocolStudies', 'isProtocolLoaded', 'isProtocolModified'}
621 argout2 = GlobalData.DataBase.iProtocol;
622 % No protocol: return empty matrix
623 if isempty(argout2) || ~isnumeric(argout2) || argout2 == 0
624 return;
625 end
626 % Check index integrity
627 if ((argout2 <= 0) || (argout2 > length(GlobalData.DataBase.ProtocolInfo))), warning('Brainstorm:InvalidIndex', 'Invalid index'), return, end
628 % Get requested protocol structure
629 argout1 = GlobalData.DataBase.(contextName)(argout2);
630
631
632 %% ==== STUDY ====
633 % Usage: [sStudy, iStudy] = bst_get('Study', StudyFileName)
634 % [sStudy, iStudy] = bst_get('Study')
635 % [sStudy, iStudy] = bst_get('Study', iStudies)
636 case 'Study'
637 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
638 return;
639 end
640 % Get list of current protocols
641 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
642 ProtocolStudies = GlobalData.DataBase.ProtocolStudies(GlobalData.DataBase.iProtocol);
643 % Get list of current protocol studies
644 if isempty(ProtocolStudies) || isempty(ProtocolInfo)
645 return;
646 end
647 % ===== PARSE INPUTS =====
648 if (nargin < 2)
649 % Call: bst_get('Study');
650 iStudies = ProtocolInfo.iStudy;
651 StudyFileName = [];
652 elseif (isnumeric(varargin{2}))
653 iStudies = varargin{2};
654 StudyFileName = [];
655 elseif (ischar(varargin{2}))
656 iStudies = [];
657 StudyFileName = strrep(varargin{2}, ProtocolInfo.STUDIES, '');
658 end
659 % Indices
660 iAnalysisStudy = -2; % CANNOT USE -1 => DISABLES SEARCH FUNCTIONS
661 iDefaultStudy = -3;
662 % Indices > 0: normal studies indiced in ProtocolStudies.Study array
663
664 % ===== GET STUDY BY INDEX =====
665 % Call: bst_get('Study', iStudies);
666 if ~isempty(iStudies)
667 argout1 = repmat(ProtocolStudies.DefaultStudy, 0);
668 % Get analysis study
669 iTargetAnalysis = find(iStudies == iAnalysisStudy);
670 if ~isempty(iTargetAnalysis)
671 argout1(iTargetAnalysis) = repmat(ProtocolStudies.AnalysisStudy, size(iTargetAnalysis));
672 argout2(iTargetAnalysis) = repmat(iAnalysisStudy, size(iTargetAnalysis));
673 end
674 % Get default study
675 iTargetDefault = find(iStudies == iDefaultStudy);
676 if ~isempty(iTargetDefault)
677 argout1(iTargetDefault) = repmat(ProtocolStudies.DefaultStudy, size(iTargetDefault));
678 argout2(iTargetDefault) = repmat(iDefaultStudy, size(iTargetDefault));
679 end
680 % Get normal studies
681 iTargetNormal = find((iStudies >= 1) & (iStudies <= length(ProtocolStudies.Study)));
682 if ~isempty(iTargetNormal)
683 argout1(iTargetNormal) = ProtocolStudies.Study(iStudies(iTargetNormal));
684 argout2(iTargetNormal) = iStudies(iTargetNormal);
685 end
686 % Error
687 if isempty(argout1)
688 GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol).iStudy = [];
689 end
690
691 % ===== GET STUDY BY FILENAME =====
692 % Call: bst_get('Study', StudyFileName);
693 elseif ~isempty(StudyFileName)
694 % NORMAL STUDY
695 iStudy = find(file_compare({ProtocolStudies.Study.FileName}, StudyFileName), 1);
696 % If a study is found : return it
697 if ~isempty(iStudy)
698 argout1 = ProtocolStudies.Study(iStudy);
699 argout2 = iStudy;
700 % DEFAULT STUDY
701 elseif ~isempty(ProtocolStudies.DefaultStudy) && file_compare({ProtocolStudies.DefaultStudy.FileName}, StudyFileName)
702 argout1 = ProtocolStudies.DefaultStudy;
703 argout2 = iDefaultStudy;
704 % ANALYSIS STUDY
705 elseif ~isempty(ProtocolStudies.AnalysisStudy) && file_compare({ProtocolStudies.AnalysisStudy.FileName}, StudyFileName)
706 argout1 = ProtocolStudies.AnalysisStudy;
707 argout2 = iAnalysisStudy;
708 end
709 else
710 return
711 end
712
713
714 %% ==== STUDY WITH SUBJECT FILE ====
715 % Usage : [sStudies, iStudies] = bst_get('StudyWithSubject', SubjectFile) : WITHOUT the system studies ('intra_subject', 'default_study')
716 % [sStudies, iStudies] = bst_get(..., 'intra_subject', 'default_study') : WITH the system studies: 'intra_subject' | 'default_study'
717 case 'StudyWithSubject'
718 % Parse inputs
719 if (nargin < 2) || ~ischar(varargin{2})
720 error('Invalid call to bst_get()');
721 end
722 if (nargin > 2)
723 IntraStudies = any(strcmpi(varargin(3:end), 'intra_subject'));
724 DefaultStudies = any(strcmpi(varargin(3:end), 'default_study'));
725 else
726 IntraStudies = 0;
727 DefaultStudies = 0;
728 end
729 SubjectFile = {varargin{2}};
730 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
731 return;
732 end
733 % Get list of current protocol description
734 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
735 ProtocolStudies = GlobalData.DataBase.ProtocolStudies(GlobalData.DataBase.iProtocol);
736 if isempty(ProtocolStudies) || isempty(ProtocolInfo)
737 return;
738 end
739
740 % Get default subject
741 sDefaultSubject = bst_get('Subject', 0);
742 % If SubjectFile is the default subject filename
743 if ~isempty(sDefaultSubject) && ~isempty(sDefaultSubject.FileName) && file_compare( SubjectFile{1}, sDefaultSubject.FileName)
744 % Get all the subjects files that use default anatomy
745 ProtocolSubjects = GlobalData.DataBase.ProtocolSubjects(GlobalData.DataBase.iProtocol);
746 iSubjectUseDefaultAnat = find([ProtocolSubjects.Subject.UseDefaultAnat]);
747 if isempty(iSubjectUseDefaultAnat)
748 return
749 end
750 SubjectFile = {ProtocolSubjects.Subject(iSubjectUseDefaultAnat).FileName};
751 % Also updates inter-subject node
752 isInterSubject = 1;
753 else
754 isInterSubject = 0;
755 end
756 % Search all the current protocol's studies
757 iStudies = [];
758 for i=1:length(SubjectFile)
759 iStudies = [iStudies, find(file_compare({ProtocolStudies.Study.BrainStormSubject}, SubjectFile{i}))];
760 end
761 % Return results
762 if ~isempty(iStudies)
763 % Remove "analysis_intra" and "default_study" studies from list
764 if ~IntraStudies
765 iStudies(strcmpi({ProtocolStudies.Study(iStudies).Name}, bst_get('DirAnalysisIntra'))) = [];
766 end
767 if ~DefaultStudies
768 iStudies(strcmpi({ProtocolStudies.Study(iStudies).Name}, bst_get('DirDefaultStudy'))) = [];
769 end
770 % Return studies
771 argout1 = ProtocolStudies.Study(iStudies);
772 argout2 = iStudies;
773 else
774 argout1 = repmat(db_template('Study'), 0);
775 argout2 = [];
776 end
777 % Add inter-subject node, if needed
778 if isInterSubject
779 [sInterStudy, iInterStudy] = bst_get('AnalysisInterStudy');
780 argout1 = [argout1, sInterStudy];
781 argout2 = [argout2, iInterStudy];
782 end
783
784
785 %% ==== STUDY WITH CONDITION PATH ====
786 % USAGE: [sStudies, iStudies] = bst_get('StudyWithCondition', ConditionPath)
787 %
788 % INPUT: ConditionPath
789 % - 'SubjectName/conditionName' : Target condition for the specified subject
790 % - 'SubjectName/@intra' : Intra-subject condition for the subject
791 % - 'SubjectName/@default_study' : Default condition for the subject (where the subject's shared files are stored)
792 % - '*/conditionName' : Target condition for all the subjects
793 % - '@inter' : Inter-subject condition
794 % - '@default_study' : Protocol's default condition (where the protocol's shared files are stored)
795
796 case 'StudyWithCondition'
797 % Parse inputs
798 if (nargin ~= 2) || ~ischar(varargin{2})
799 error('Invalid call to bst_get()');
800 end
801 ConditionPath = varargin{2};
802 % Get list of current protocol description
803 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
804 ProtocolStudies = GlobalData.DataBase.ProtocolStudies(GlobalData.DataBase.iProtocol);
805 if isempty(ProtocolStudies) || isempty(ProtocolInfo)
806 return;
807 end
808
809 % ConditionPath = @inter
810 if strcmpi(ConditionPath, '@inter')
811 iStudy = -2;
812 argout2 = iStudy;
813 argout1 = bst_get('Study', iStudy);
814 % ConditionPath = @default_study
815 elseif strcmpi(ConditionPath, '@default_study')
816 iStudy = -3;
817 argout2 = iStudy;
818 argout1 = bst_get('Study', iStudy);
819 % ConditionPath = SubjectName/ConditionName
820 else
821 % Get subject and condition names
822 condSplit = str_split(ConditionPath);
823 if (length(condSplit) ~= 2)
824 error('Invalid condition path.');
825 end
826 SubjectName = condSplit{1};
827 ConditionName = condSplit{2};
828
829 % If first element is '*', search for condition in all the studies
830 if (SubjectName(1) == '*')
831 iStudies = 1:length(ProtocolStudies.Study);
832 % Else : search for condition only in studies that are linked to the subject specified in the ConditionPath
833 else
834 % Get subject
835 sSubject = bst_get('Subject', SubjectName, 1);
836 if isempty(sSubject)
837 return;
838 end
839 iStudies = find(file_compare({ProtocolStudies.Study.BrainStormSubject}, sSubject.FileName));
840 end
841 % Nothing to search
842 if isempty(iStudies)
843 return
844 end
845
846 % Search all the current protocol's studies
847 iStudies = iStudies(cellfun(@(c)isequal(ConditionName, c), [ProtocolStudies.Study(iStudies).Condition]));
848 % Return results
849 if ~isempty(iStudies)
850 % Remove "analysis_intra" and "default_study" studies from list
851 if ~strcmpi(ConditionName, '@intra')
852 iStudies(strcmpi({ProtocolStudies.Study(iStudies).Condition}, bst_get('DirAnalysisIntra'))) = [];
853 end
854 if ~strcmpi(ConditionName, '@default_study')
855 iStudies(strcmpi({ProtocolStudies.Study(iStudies).Condition}, bst_get('DirDefaultStudy'))) = [];
856 end
857 % Sort by subject
858 if (length(iStudies) > 1)
859 SubjNameList = cell(1,length(iStudies));
860 % For each study, get subject name
861 for i = 1:length(iStudies)
862 sSubject = bst_get('Subject', ProtocolStudies.Study(iStudies(i)).BrainStormSubject);
863 SubjNameList{i} = sSubject.Name;
864 end
865 % Sort subjects names
866 [sortSubjList, iSort] = sort(SubjNameList);
867 % Apply same sorting to studies
868 iStudies = iStudies(iSort);
869 end
870 % Return studies
871 argout1 = ProtocolStudies.Study(iStudies);
872 argout2 = iStudies;
873 else
874 argout1 = repmat(db_template('Study'), 0);
875 argout2 = [];
876 end
877 end
878
879 %% ==== CHANNEL STUDIES WITH SUBJECT ====
880 % Usage: iStudies = bst_get('ChannelStudiesWithSubject', iSubjects, 'NoIntra')
881 case 'ChannelStudiesWithSubject'
882 % Parse inputs
883 if (nargin >= 2) && isnumeric(varargin{2})
884 iSubjects = varargin{2};
885 else
886 error('Invalid call to bst_get()');
887 end
888 if (nargin == 3) && strcmpi(varargin{3}, 'NoIntra')
889 NoIntra = 1;
890 else
891 NoIntra = 0;
892 end
893 % Process all subjects
894 iStudies = [];
895 for i=1:length(iSubjects)
896 iSubject = iSubjects(i);
897 sSubject = bst_get('Subject', iSubject, 1);
898 % No subject: error
899 if isempty(sSubject)
900 continue
901 % If subject uses default channel file
902 elseif (sSubject.UseDefaultChannel ~= 0)
903 % Get default study for this subject
904 [tmp___, iStudiesNew] = bst_get('DefaultStudy', iSubject);
905 iStudies = [iStudies, iStudiesNew];
906 % Else: get all the studies belonging to this subject
907 else
908 if NoIntra
909 [tmp___, iStudiesNew] = bst_get('StudyWithSubject', sSubject.FileName);
910 else
911 [tmp___, iStudiesNew] = bst_get('StudyWithSubject', sSubject.FileName, 'intra_subject');
912 end
913 iStudies = [iStudies, iStudiesNew];
914 end
915 end
916 argout1 = iStudies;
917
918
919 %% ==== STUDIES COUNT ====
920 % Usage: [nbStudies] = bst_get('StudyCount')
921 case 'StudyCount'
922 % Nothing
923 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
924 argout1 = 0;
925 return;
926 end
927 % Get list of current protocol studies
928 ProtocolStudies = GlobalData.DataBase.ProtocolStudies(GlobalData.DataBase.iProtocol);
929 argout1 = length(ProtocolStudies.Study);
930
931 %% ==== SUBJECTS COUNT ====
932 % Usage: [nbSubjects] = bst_get('SubjectCount')
933 case 'SubjectCount'
934 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
935 argout1 = 0;
936 return;
937 end
938 % Get list of current protocol studies
939 ProtocolSubjects = GlobalData.DataBase.ProtocolSubjects(GlobalData.DataBase.iProtocol);
940 argout1 = length(ProtocolSubjects.Subject);
941
942 %% ==== NORMALIZED SUBJECT ====
943 case 'NormalizedSubject'
944 % Get normalized subject name
945 normSubjName = 'Group_analysis';
946 % Try to get normalized subject
947 [sNormSubj, iNormSubj] = bst_get('Subject', normSubjName, 0);
948 % If normalized subject does not exist: create it
949 if isempty(sNormSubj)
950 % Always use default anatomy
951 UseDefaultAnat = 1;
952 % If all the subjects use a global channel file: Use global default as well
953 if ~isempty(GlobalData.DataBase.ProtocolSubjects(GlobalData.DataBase.iProtocol).Subject) && ...
954 all([GlobalData.DataBase.ProtocolSubjects(GlobalData.DataBase.iProtocol).Subject.UseDefaultChannel] == 2)
955 UseDefaultChannel = 2;
956 else
957 UseDefaultChannel = 1;
958 end
959 % Create subject
960 [sNormSubj, iNormSubj] = db_add_subject(normSubjName, [], UseDefaultAnat, UseDefaultChannel);
961 % Get full subject structure
962 [sNormSubj, iNormSubj] = bst_get('Subject', normSubjName, 0);
963 end
964 argout1 = sNormSubj;
965 argout2 = iNormSubj;
966
967
968 %% ==== ANALYSIS STUDY (INTRA) ====
969 % Usage: [sAnalStudy, iAnalStudy] = bst_get('AnalysisIntraStudy', iSubject)
970 case 'AnalysisIntraStudy'
971 % Parse inputs
972 if (nargin == 2)
973 iSubject = varargin{2};
974 else
975 error('Invalid call to bst_get()');
976 end
977 % Get subject
978 sSubject = bst_get('Subject', iSubject, 1);
979 % Get studies related to subject
980 [sSubjStudies, iSubjStudies] = bst_get('StudyWithSubject', sSubject.FileName, 'intra_subject');
981 % Look for the 'AnalysisIntra' study
982 iFound = find(cellfun(@(c)ismember(bst_get('DirAnalysisIntra'), c), {sSubjStudies.Condition}));
983 iAnalStudy = iSubjStudies(iFound);
984 sAnalStudy = sSubjStudies(iFound);
985 % Return found structure
986 argout1 = sAnalStudy;
987 argout2 = iAnalStudy;
988
989
990 %% ==== ANALYSIS STUDY (INTER) ====
991 % Usage: [sAnalStudyInter, iAnalStudyInter] = bst_get('AnalysisInterStudy')
992 case 'AnalysisInterStudy'
993 iAnalStudyInter = -2;
994 [argout1, argout2] = bst_get('Study', iAnalStudyInter);
995
996
997 %% ==== DEFAULT STUDY ====
998 % Usage: [sDefaulStudy, iDefaultStudy] = bst_get('DefaultStudy', iSubject)
999 % [sDefaulStudy, iDefaultStudy] = bst_get('DefaultStudy') : iSubject=0
1000 % [sDefaulStudy, iDefaultStudy] = bst_get('DefaultStudy', SubjectFile)
1001 case 'DefaultStudy'
1002 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
1003 return;
1004 end
1005 % Parse inputs
1006 if (nargin == 1)
1007 iSubject = 0;
1008 elseif (nargin == 2) && isnumeric(varargin{2})
1009 iSubject = varargin{2};
1010 elseif (nargin == 2) && ischar(varargin{2})
1011 SubjectFile = varargin{2};
1012 % Get subject attached to study
1013 [sSubject, iSubject] = bst_get('Subject', SubjectFile, 1);
1014 if isempty(sSubject) || ~sSubject.UseDefaultChannel
1015 return;
1016 end
1017 else
1018 error('Invalid call to bst_get()');
1019 end
1020 % === DEFAULT SUBJECT ===
1021 % => Return global default study
1022 if (iSubject == 0)
1023 % Get protocol's studies
1024 ProtocolStudies = GlobalData.DataBase.ProtocolStudies(GlobalData.DataBase.iProtocol);
1025 % Return Global default study
1026 argout1 = ProtocolStudies.DefaultStudy;
1027 argout2 = -3;
1028 % === NORMAL SUBJECT ===
1029 else
1030 % Get subject
1031 sSubject = bst_get('Subject', iSubject, 1);
1032 % === GLOBAL DEFAULT STUDY ===
1033 if sSubject.UseDefaultChannel == 2
1034 % Get protocol's studies
1035 ProtocolStudies = GlobalData.DataBase.ProtocolStudies(GlobalData.DataBase.iProtocol);
1036 % Return Global default study
1037 argout1 = ProtocolStudies.DefaultStudy;
1038 argout2 = -3;
1039 % === SUBJECT'S DEFAULT STUDY ===
1040 elseif sSubject.UseDefaultChannel == 1
1041 % Get studies related to subject
1042 [sSubjStudies, iSubjStudies] = bst_get('StudyWithSubject', sSubject.FileName, 'default_study');
1043 % Look for the 'DefaultStudy' study
1044 iFound = find(cellfun(@(c)ismember(bst_get('DirDefaultStudy'), c), {sSubjStudies.Condition}));
1045 iDefaultStudy = iSubjStudies(iFound);
1046 sDefaultStudy = sSubjStudies(iFound);
1047 % Return found structure
1048 argout1 = sDefaultStudy;
1049 argout2 = iDefaultStudy;
1050 end
1051 end
1052
1053
1054
1055 %% ==== SUBJECT ====
1056 % Usage : [sSubject, iSubject] = bst_get('Subject', iSubject, isRaw)
1057 % [sSubject, iSubject] = bst_get('Subject', SubjectFileName, isRaw);
1058 % [sSubject, iSubject] = bst_get('Subject', SubjectName, isRaw);
1059 % [sSubject, iSubject] = bst_get('Subject');
1060 % If isRaw is set: force to return the real brainstormsubject description
1061 % (ignoring wether it uses protocol's default anatomy or not)
1062 case 'Subject'
1063 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
1064 return;
1065 end
1066 % Get list of current protocol subjects
1067 ProtocolSubjects = GlobalData.DataBase.ProtocolSubjects(GlobalData.DataBase.iProtocol);
1068 sSubject = [];
1069 SubjectName = [];
1070 SubjectFileName = [];
1071 % ISRAW parameter
1072 if (nargin < 3)
1073 isRaw = 0;
1074 else
1075 isRaw = varargin{3};
1076 end
1077 % Call: bst_get('subject', iSubject, isRaw);
1078 if (nargin >= 2) && isnumeric(varargin{2})
1079 iSubject = varargin{2};
1080 if (iSubject > length(ProtocolSubjects.Subject))
1081 error('Invalid subject indice.');
1082 end
1083 % If required subject is default subject (iSubject = 0)
1084 if (iSubject == 0)
1085 % Default subject available
1086 if ~isempty(ProtocolSubjects.DefaultSubject)
1087 sSubject = ProtocolSubjects.DefaultSubject;
1088 % Default subject not available
1089 else
1090 return
1091 end
1092 % Normal subject
1093 else
1094 sSubject = ProtocolSubjects.Subject(iSubject);
1095 end
1096
1097 % Call: bst_get('subject', SubjectFileName, isRaw);
1098 % Call: bst_get('subject', SubjectName, isRaw);
1099 elseif (nargin >= 2) && isempty(varargin{2})
1100 % If study name is empty: use DefaultSubject
1101 SubjectFileName = ProtocolSubjects.DefaultSubject.FileName;
1102 elseif (nargin >= 2) && (ischar(varargin{2}))
1103 [fName, fBase, fExt] = bst_fileparts(varargin{2});
1104 % Argument is a Matlab .mat filename
1105 if strcmpi(fExt, '.mat')
1106 SubjectFileName = varargin{2};
1107 % Else : assume argument is a directory
1108 else
1109 SubjectName = file_standardize(varargin{2});
1110 end
1111
1112 % Call: bst_get('subject'); => looking for current subject
1113 elseif (nargin < 2)
1114 % Get current subject filename in current study
1115 sStudy = bst_get('Study');
1116 if isempty(sStudy)
1117 return
1118 end
1119 SubjectFileName = sStudy.BrainStormSubject;
1120 % If study's subject is not defined, get DefaultSubject
1121 if isempty(SubjectFileName) && ~isempty(ProtocolSubjects.DefaultSubject)
1122 SubjectFileName = ProtocolSubjects.DefaultSubject.FileName;
1123 end
1124 else
1125 error('Invalid call to bst_get()');
1126 end
1127
1128 % If Subject is defined by its filename/name
1129 if isempty(sSubject)
1130 % Look in Default Subject
1131 if ~isempty(ProtocolSubjects.DefaultSubject) && (file_compare(ProtocolSubjects.DefaultSubject.FileName, SubjectFileName) ...
1132 || strcmpi(ProtocolSubjects.DefaultSubject.Name, SubjectName))
1133 sSubject = ProtocolSubjects.DefaultSubject;
1134 iSubject = 0;
1135 % If not found : find target subject file name in normal subjects
1136 elseif ~isempty(SubjectFileName)
1137 iSubject = find(file_compare({ProtocolSubjects.Subject.FileName}, SubjectFileName), 1);
1138 sSubject = ProtocolSubjects.Subject(iSubject);
1139 elseif ~isempty(SubjectName)
1140 iSubject = find(file_compare({ProtocolSubjects.Subject.Name}, SubjectName), 1);
1141 sSubject = ProtocolSubjects.Subject(iSubject);
1142 else
1143 error('Subject name not specified.');
1144 end
1145 end
1146
1147 % Return found subject
1148 if ~isempty(iSubject) && ~isempty(sSubject)
1149 % If subject uses default subject
1150 if sSubject.UseDefaultAnat && ~isRaw && ~isempty(ProtocolSubjects.DefaultSubject) && ~isempty(ProtocolSubjects.DefaultSubject.FileName)
1151 % Return default subject (WITH REAL SUBJECT'S NAME)
1152 argout1 = ProtocolSubjects.DefaultSubject;
1153 argout1.Name = sSubject.Name;
1154 argout1.UseDefaultAnat = sSubject.UseDefaultAnat;
1155 argout1.UseDefaultChannel = sSubject.UseDefaultChannel;
1156 argout2 = iSubject;
1157 % Else, return found subject
1158 else
1159 argout1 = sSubject;
1160 argout2 = iSubject;
1161 end
1162 end
1163
1164
1165 %% ==== SURFACE FILE ====
1166 % Usage : [sSubject, iSubject, iSurface] = bst_get('SurfaceFile', SurfaceFile)
1167 case 'SurfaceFile'
1168 % No protocol in database
1169 if isempty(GlobalData) || isempty(GlobalData.DataBase) || isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
1170 return;
1171 end
1172 % Get list of current protocol subjects
1173 ProtocolSubjects = GlobalData.DataBase.ProtocolSubjects(GlobalData.DataBase.iProtocol);
1174 if isempty(ProtocolSubjects)
1175 return
1176 end;
1177
1178 % Parse inputs
1179 if (nargin == 2)
1180 SurfaceFile = varargin{2};
1181 else
1182 error('Invalid call to bst_get().');
1183 end
1184
1185 % Remove SUBJECTS path from SurfaceFile
1186 SurfaceFile = file_short(SurfaceFile);
1187 % Look for surface file in DefaultSubject
1188 if ~isempty(ProtocolSubjects.DefaultSubject)
1189 % Find the first surface that matches the SurfaceFile
1190 iSurface = find(file_compare(SurfaceFile, {ProtocolSubjects.DefaultSubject.Surface.FileName}), 1);
1191 % If a surface was found in default subject : return it
1192 if ~isempty(iSurface)
1193 argout1 = ProtocolSubjects.DefaultSubject;
1194 argout2 = 0;
1195 argout3 = iSurface;
1196 return
1197 end
1198 end
1199 % Look for surface file in all the surfaces of all subjects
1200 for iSubj = 1:length(ProtocolSubjects.Subject)
1201 % Find the first surface that matches the SurfaceFile
1202 iSurface = find(file_compare(SurfaceFile, {ProtocolSubjects.Subject(iSubj).Surface.FileName}), 1);
1203 % If a surface was found in current subject : return it
1204 if ~isempty(iSurface)
1205 argout1 = ProtocolSubjects.Subject(iSubj);
1206 argout2 = iSubj;
1207 argout3 = iSurface;
1208 return
1209 end
1210 end
1211
1212
1213 %% ==== SURFACE FILE BY TYPE ====
1214 % Usage : [sSurface, iSurface] = bst_get('SurfaceFileByType', iSubject, SurfaceType)
1215 % [sSurface, iSurface] = bst_get('SurfaceFileByType', SubjectFile, SurfaceType)
1216 % [sSurface, iSurface] = bst_get('SurfaceFileByType', SurfaceFile, SurfaceType)
1217 % [sSurface, iSurface] = bst_get('SurfaceFileByType', MriFile, SurfaceType)
1218 % [sSurface, iSurface] = bst_get('SurfaceFileByType', ..., SurfaceType, isDefaultOnly)
1219 case 'SurfaceFileByType'
1220 % By default: return only the default surfaces of the category
1221 if (nargin < 4)
1222 isDefaultOnly = 1;
1223 else
1224 isDefaultOnly = varargin{4};
1225 end
1226 % Get subject
1227 if isempty(varargin{2})
1228 % Get default subject
1229 sSubject = bst_get('Subject', 0);
1230 elseif ischar(varargin{2})
1231 FileName = varargin{2};
1232 sSubject = bst_get('AnyFile', FileName);
1233 else
1234 iSubject = varargin{2};
1235 sSubject = bst_get('Subject', iSubject);
1236 end
1237 % Error handling
1238 if isempty(sSubject)
1239 disp('BST> Warning: Subject not found.');
1240 return;
1241 elseif isempty(sSubject.Surface)
1242 return;
1243 end
1244 SurfaceType = varargin{3};
1245
1246 % === RETURN ONLY DEFAULTS ===
1247 if isDefaultOnly
1248 % Look for required surface type
1249 field = ['i' SurfaceType];
1250 if ~isfield(sSubject, field) || isempty(sSubject.(field))
1251 return
1252 end
1253 argout1 = sSubject.Surface(sSubject.(field));
1254 argout2 = sSubject.(field);
1255 % === RETURN ALL THE SURFACES ===
1256 else
1257 % Build the list of tagged surfaces
1258 fileTag = ['_' lower(SurfaceType)];
1259 iTargetList = find(cellfun(@(c)~isempty(strfind(c, fileTag)), {sSubject.Surface.FileName}));
1260 % Put the default cortex on top of the list
1261 iDefaults = intersect([sSubject.iCortex, sSubject.iScalp, sSubject.iInnerSkull, sSubject.iOuterSkull, sSubject.iFibers, sSubject.iFEM], iTargetList);
1262 if ~isempty(iDefaults)
1263 iTargetList = [iDefaults, setdiff(iTargetList, iDefaults)];
1264 end
1265 % Return all cortex surfaces
1266 argout1 = sSubject.Surface(iTargetList);
1267 argout2 = iTargetList;
1268 end
1269
1270
1271 %% ==== MRI FILE ====
1272 % Usage : [sSubject, iSubject, iMri] = bst_get('MriFile', MriFile)
1273 case 'MriFile'
1274 % No protocol in database
1275 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
1276 return;
1277 end
1278 % Get list of current protocol subjects
1279 ProtocolSubjects = GlobalData.DataBase.ProtocolSubjects(GlobalData.DataBase.iProtocol);
1280 if isempty(ProtocolSubjects)
1281 return
1282 end
1283
1284 % Parse inputs
1285 if (nargin == 2)
1286 MriFile = varargin{2};
1287 else
1288 error('Invalid call to bst_get().');
1289 end
1290
1291 % Remove SUBJECTS path from MriFile
1292 MriFile = file_short(MriFile);
1293 % Look for MRI file in DefaultSubject
1294 if ~isempty(ProtocolSubjects.DefaultSubject)
1295 % Find the first MRI that matches the MriFile
1296 iMri = find(file_compare(MriFile, {ProtocolSubjects.DefaultSubject.Anatomy.FileName}), 1);
1297 % If a MRI was found in default subject : return it
1298 if ~isempty(iMri)
1299 argout1 = ProtocolSubjects.DefaultSubject;
1300 argout2 = 0;
1301 argout3 = iMri;
1302 return
1303 end
1304 end
1305 % Look for MRI file in all the MRIs of all subjects
1306 for iSubj = 1:length(ProtocolSubjects.Subject)
1307 % Find the first MRI that matches the MriFile
1308 iMri = find(file_compare(MriFile, {ProtocolSubjects.Subject(iSubj).Anatomy.FileName}), 1);
1309 % If a MRI was found in current subject : return it
1310 if ~isempty(iMri)
1311 argout1 = ProtocolSubjects.Subject(iSubj);
1312 argout2 = iSubj;
1313 argout3 = iMri;
1314 return
1315 end
1316 end
1317
1318
1319 %% ==== CHANNEL FILE ====
1320 % Usage: [sStudy, iStudy, iChannel] = bst_get('ChannelFile', ChannelFile)
1321 case 'ChannelFile'
1322 % No protocol in database
1323 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
1324 return;
1325 end
1326 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
1327 % Parse inputs
1328 if (nargin == 2)
1329 ChannelFile = varargin{2};
1330 ChannelFile = strrep(ChannelFile, ProtocolInfo.STUDIES, '');
1331 else
1332 error('Invalid call to bst_get().');
1333 end
1334 % Look for Channel file in all the surfaces of all subjects
1335 [argout1, argout2, argout3] = findFileInStudies('Channel', 'FileName', ChannelFile);
1336
1337
1338 %% ==== CHANNEL FILE FOR STUDY ====
1339 % Usage: [ChannelFile, sStudy, iStudy] = bst_get('ChannelFileForStudy', StudyFile/DataFile)
1340 case 'ChannelFileForStudy'
1341 % Parse inputs
1342 if (nargin == 2)
1343 StudyFile = varargin{2};
1344 else
1345 error('Invalid call to bst_get().');
1346 end
1347 % Get study in database
1348 [sStudy, iStudy] = bst_get('Study', StudyFile);
1349 % If data file instead of Study file
1350 if isempty(sStudy)
1351 [sStudy, iStudy] = bst_get('AnyFile', StudyFile);
1352 end
1353 sChannel = bst_get('ChannelForStudy', iStudy);
1354 if ~isempty(sChannel)
1355 argout1 = sChannel.FileName;
1356 argout2 = sStudy;
1357 argout3 = iStudy;
1358 else
1359 argout1 = [];
1360 end
1361
1362
1363 %% ==== CHANNEL STRUCT FOR STUDY ====
1364 % Usage: [sChannel, iChanStudy] = bst_get('ChannelForStudy', iStudies)
1365 case 'ChannelForStudy'
1366 % Parse inputs
1367 if (nargin == 2)
1368 iStudies = varargin{2};
1369 else
1370 error('Invalid call to bst_get().');
1371 end
1372 iChanStudies = [];
1373 sListChannel = [];
1374 for i = 1:length(iStudies)
1375 % Get study
1376 iStudy = iStudies(i);
1377 sStudy = bst_get('Study', iStudy);
1378 if isempty(sStudy)
1379 continue;
1380 end
1381 iChanStudy = iStudy;
1382 % === Analysis-Inter node ===
1383 iAnalysisInter = -2;
1384 iGlobalDefaultStudy = -3;
1385 if (iStudy == iAnalysisInter)
1386 % If no channel file is defined in 'Analysis-intra' node: look in
1387 if isempty(sStudy.Channel)
1388 % Get global default study
1389 sStudy = bst_get('Study', iGlobalDefaultStudy);
1390 iChanStudy = iGlobalDefaultStudy;
1391 end
1392 % === All other nodes ===
1393 else
1394 % Get subject attached to study
1395 [sSubject, iSubject] = bst_get('Subject', sStudy.BrainStormSubject, 1);
1396 if isempty(sSubject)
1397 return;
1398 end
1399 % Subject uses default channel/headmodel
1400 if (sSubject.UseDefaultChannel ~= 0)
1401 [sStudy, iChanStudy] = bst_get('DefaultStudy', iSubject);
1402 if isempty(sStudy)
1403 return
1404 end
1405 end
1406 end
1407 iChanStudies = [iChanStudies, iChanStudy];
1408 sListChannel = [sListChannel, sStudy.Channel];
1409 end
1410 % Return Channel structure
1411 argout1 = sListChannel;
1412 argout2 = iChanStudies;
1413
1414 %% ==== CHANNEL MODALITIES =====
1415 % Usage: [Modalities, DispMod, DefMod] = bst_get('ChannelModalities', ChannelFile)
1416 % [Modalities, DispMod, DefMod] = bst_get('ChannelModalities', DataFile/ResultsFile/TimefreqFile...)
1417 case 'ChannelModalities'
1418 % Get input file
1419 [sStudy, iStudy, iItem, DataType, sItem] = bst_get('AnyFile', varargin{2});
1420 % If channel file
1421 if strcmpi(DataType, 'channel')
1422 sChannel = sItem;
1423 else
1424 sChannel = bst_get('ChannelForStudy', iStudy);
1425 end
1426 % Return modalities
1427 if ~isempty(sChannel)
1428 % Get all modalities
1429 if ~isempty(sChannel.DisplayableSensorTypes)
1430 % Return the displayable sensors on top of the list
1431 argout1 = cat(2, sChannel.DisplayableSensorTypes, setdiff(sChannel.Modalities, sChannel.DisplayableSensorTypes));
1432 else
1433 argout1 = sChannel.Modalities;
1434 end
1435 % Get only sensors that have spatial representation
1436 argout2 = sChannel.DisplayableSensorTypes;
1437 % Default candidates
1438 if ~isempty(argout2)
1439 defList = argout2;
1440 else
1441 defList = argout1;
1442 end
1443 if isempty(defList)
1444 return;
1445 end
1446 % Remove EDF and BDF from the default list
1447 defList = setdiff(defList, {'EDF','BDF','KDF'});
1448 % Get default modality
1449 if ismember('SEEG', defList)
1450 argout3 = 'SEEG';
1451 elseif ismember('ECOG', defList)
1452 argout3 = 'ECOG';
1453 elseif any(ismember({'MEG','MEG GRAD','MEG MAG'}, defList))
1454 argout3 = 'MEG';
1455 elseif ismember('EEG', defList)
1456 argout3 = 'EEG';
1457 elseif ismember('NIRS', defList)
1458 argout3 = 'NIRS';
1459 else
1460 argout3 = defList{1};
1461 end
1462 % Place the default on top of the lists
1463 if ismember(argout3, argout1)
1464 argout1(strcmpi(argout1, argout3)) = [];
1465 argout1 = cat(2, argout3, argout1);
1466 end
1467 if ismember(argout3, argout2)
1468 argout2(strcmpi(argout2, argout3)) = [];
1469 argout2 = cat(2, argout3, argout2);
1470 end
1471 else
1472 argout1 = [];
1473 end
1474
1475
1476 %% ==== TIMEFREQ DISPLAY MODALITIES ====
1477 % Usage: DisplayMod = bst_get('TimefreqDisplayModalities', TimefreqFile)
1478 case 'TimefreqDisplayModalities'
1479 TimefreqFile = varargin{2};
1480 % Load sensor names from file
1481 TimefreqMat = in_bst_timefreq(TimefreqFile, 0, 'RowNames');
1482 % Get channel file
1483 ChannelFile = bst_get('ChannelFileForStudy', TimefreqFile);
1484 if isempty(ChannelFile)
1485 return;
1486 end
1487 % Load channel file
1488 ChannelMat = load(file_fullpath(ChannelFile), 'Channel');
1489 % Get channels that are present in the file
1490 [tmp__,I,J] = intersect({ChannelMat.Channel.Name}, TimefreqMat.RowNames);
1491 FileMod = unique({ChannelMat.Channel(I).Type});
1492 % Check if only one type of gradiometer is selected
1493 if isequal(FileMod, {'MEG GRAD'}) && all(cellfun(@(c)(c(end)=='2'), {ChannelMat.Channel(I).Name}))
1494 argout1 = {'MEG GRAD2'};
1495 elseif isequal(FileMod, {'MEG GRAD'}) && all(cellfun(@(c)(c(end)=='3'), {ChannelMat.Channel(I).Name}))
1496 argout1 = {'MEG GRAD3'};
1497 % Keep only the modalities that can be displayed (as topography)
1498 else
1499 argout1 = intersect(FileMod, {'MEG','MEG GRAD','MEG MAG','EEG','ECOG','SEEG','NIRS'});
1500 end
1501
1502
1503 %% ==== CHANNEL DEVICE ====
1504 % Usage: Device = bst_get('ChannelDevice', ChannelFile)
1505 case 'ChannelDevice'
1506 ChannelFile = varargin{2};
1507 if ~isempty(strfind(ChannelFile, 'vectorview306'))
1508 Device = 'Vectorview306';
1509 elseif ~isempty(strfind(ChannelFile, 'ctf_acc1'))
1510 Device = 'CTF';
1511 elseif ~isempty(strfind(ChannelFile, '4d_acc1'))
1512 Device = '4D';
1513 elseif ~isempty(strfind(ChannelFile, 'babysquid'))
1514 Device = 'BabySQUID';
1515 elseif ~isempty(strfind(ChannelFile, 'babymeg'))
1516 Device = 'BabyMEG';
1517 elseif ~isempty(strfind(ChannelFile, 'kit'))
1518 Device = 'KIT';
1519 elseif ~isempty(strfind(ChannelFile, 'ricoh'))
1520 Device = 'RICOH';
1521 elseif ~isempty(strfind(ChannelFile, 'kriss'))
1522 Device = 'KRISS';
1523 elseif ~isempty(strfind(ChannelFile, 'nirsbrs'))
1524 Device = 'NIRS-BRS';
1525 else
1526 Device = '';
1527 end
1528 argout1 = Device;
1529
1530
1531 %% ==== HEADMODEL STRUCT FOR STUDY ====
1532 % Usage: [sHeadModel] = bst_get('HeadModelForStudy', iStudy)
1533 case 'HeadModelForStudy'
1534 % Parse inputs
1535 if (nargin == 2)
1536 iStudy = varargin{2};
1537 else
1538 error('Invalid call to bst_get().');
1539 end
1540 % Get study
1541 sStudy = bst_get('Study', iStudy);
1542 % === Analysis-Inter node ===
1543 iAnalysisInter = -2;
1544 iGlobalDefaultStudy = -3;
1545 if (iStudy == iAnalysisInter)
1546 % If no channel file is defined in 'Analysis-intra' node: look in
1547 if isempty(sStudy.iHeadModel)
1548 % Get global default study
1549 sStudy = bst_get('Study', iGlobalDefaultStudy);
1550 end
1551 % === All other nodes ===
1552 else
1553 % Get subject attached to study
1554 [sSubject, iSubject] = bst_get('Subject', sStudy.BrainStormSubject, 1);
1555 if isempty(sSubject)
1556 return;
1557 end
1558 % Subject uses default channel/headmodel
1559 if (sSubject.UseDefaultChannel ~= 0)
1560 sStudy = bst_get('DefaultStudy', iSubject);
1561 if isempty(sStudy)
1562 return
1563 end
1564 end
1565 end
1566 % Return HeadModel structure
1567 if ~isempty(sStudy.iHeadModel)
1568 argout1 = sStudy.HeadModel(sStudy.iHeadModel(1));
1569 else
1570 argout1 = [];
1571 end
1572
1573
1574 %% ==== HEADMODEL FILE ====
1575 % Usage: [sStudy, iStudy, iHeadModel] = bst_get('HeadModelFile', HeadModelFile, iStudies)
1576 % [sStudy, iStudy, iHeadModel] = bst_get('HeadModelFile', HeadModelFile)
1577 case 'HeadModelFile'
1578 % No protocol in database
1579 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
1580 return;
1581 end
1582 % Input #2: HeadModelFile
1583 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
1584 HeadModelFile = varargin{2};
1585 HeadModelFile = strrep(HeadModelFile, ProtocolInfo.STUDIES, '');
1586 % Input #3: iStudies
1587 if (nargin < 3)
1588 iStudies = [];
1589 else
1590 iStudies = varargin{3};
1591 end
1592 % Look for surface file in all the surfaces of all subjects
1593 [argout1, argout2, argout3] = findFileInStudies('HeadModel', 'FileName', HeadModelFile, iStudies);
1594
1595 %% ==== NOISECOV FILE ====
1596 % Usage: [sStudy, iStudy, iNoiseCov] = bst_get('NoiseCovFile', NoiseCovFile, iStudies)
1597 % [sStudy, iStudy, iNoiseCov] = bst_get('NoiseCovFile', NoiseCovFile)
1598 % Usage: [sStudy, iStudy, iNoiseCov] = bst_get('DataCovFile', NoiseCovFile, iStudies)
1599 % [sStudy, iStudy, iNoiseCov] = bst_get('DataCovFile', NoiseCovFile)
1600 case {'NoiseCovFile', 'DataCovFile'}
1601 % No protocol in database
1602 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
1603 return;
1604 end
1605 % Input #2: NoiseCovFile
1606 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
1607 NoiseCovFile = varargin{2};
1608 NoiseCovFile = strrep(NoiseCovFile, ProtocolInfo.STUDIES, '');
1609 % Input #3: iStudies
1610 if (nargin < 3)
1611 iStudies = [];
1612 else
1613 iStudies = varargin{3};
1614 end
1615 % Look for surface file in all the surfaces of all subjects
1616 [argout1, argout2, argout3] = findFileInStudies('NoiseCov', 'FileName', NoiseCovFile, iStudies);
1617
1618
1619 %% ==== DATA FILE ====
1620 % Usage: [sStudy, iStudy, iData] = bst_get('DataFile', DataFile, iStudies)
1621 % [sStudy, iStudy, iData] = bst_get('DataFile', DataFile)
1622 case 'DataFile'
1623 % No protocol in database
1624 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
1625 return;
1626 end
1627 % Input #2: DataFile
1628 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
1629 DataFile = varargin{2};
1630 DataFile = strrep(DataFile, ProtocolInfo.STUDIES, '');
1631 % Input #3: iStudies
1632 if (nargin < 3)
1633 iStudies = [];
1634 else
1635 iStudies = varargin{3};
1636 end
1637 % Look for file in all the studies
1638 [argout1, argout2, argout3] = findFileInStudies('Data', 'FileName', DataFile, iStudies);
1639
1640
1641 %% ==== DATA FOR DATA LIST ====
1642 % Usage: [iFoundData] = bst_get('DataForDataList', iStudy, DataListName)
1643 case 'DataForDataList'
1644 iStudy = varargin{2};
1645 DataListName = varargin{3};
1646 % Get study structure
1647 sStudy = bst_get('Study', iStudy);
1648 % Get all the data files held by this datalist
1649 listComments = cellfun(@(c)deblank(str_remove_parenth(c)), {sStudy.Data.Comment}, 'UniformOutput', 0);
1650 iFoundData = find(strcmp(listComments, DataListName));
1651 % Return found data files
1652 argout1 = iFoundData;
1653
1654 %% ==== MATRIX FOR MATRIX LIST ====
1655 % Usage: [iFoundMatrix] = bst_get('MatrixForMatrixList', iStudy, MatrixListName)
1656 case 'MatrixForMatrixList'
1657 iStudy = varargin{2};
1658 MatrixListName = varargin{3};
1659 % Get study structure
1660 sStudy = bst_get('Study', iStudy);
1661 % Get all the matrix files held by this datalist
1662 listComments = cellfun(@(c)deblank(str_remove_parenth(c)), {sStudy.Matrix.Comment}, 'UniformOutput', 0);
1663 iFoundMatrix = find(strcmp(listComments, MatrixListName));
1664 % Return found matrix files
1665 argout1 = iFoundMatrix;
1666
1667
1668 %% ==== DATA FOR STUDY (INCLUDING SHARED STUDIES) ====
1669 % Usage: [iStudies, iDatas] = bst_get('DataForStudy', iStudy)
1670 case 'DataForStudy'
1671 % Get target study
1672 iStudy = varargin{2};
1673 sStudy = bst_get('Study', iStudy);
1674 isDefaultStudy = strcmpi(sStudy.Name, bst_get('DirDefaultStudy'));
1675 isGlobalDefault = (iStudy == -3);
1676
1677 % If study is the global default study
1678 sStudies = [];
1679 iStudies = [];
1680 if isGlobalDefault
1681 % Get all the subjects of the protocol
1682 nbSubjects = bst_get('SubjectCount');
1683 for iSubject = 1:nbSubjects
1684 sSubject = bst_get('Subject', iSubject, 1);
1685 if sSubject.UseDefaultChannel
1686 [tmp_sStudies, tmp_iStudies] = bst_get('StudyWithSubject', sSubject.FileName);
1687 sStudies = [sStudies, tmp_sStudies];
1688 iStudies = [iStudies, tmp_iStudies];
1689 end
1690 end
1691 % Else, if study is a subject's default study (ie. channel file is shared by all studies of one subject)
1692 elseif isDefaultStudy
1693 % Get all the subject's studies
1694 [sStudies, iStudies] = bst_get('StudyWithSubject', sStudy.BrainStormSubject, 'intra_subject', 'default_study');
1695 else
1696 % Normal: one channel per condition
1697 sStudies = sStudy;
1698 iStudies = iStudy;
1699 end
1700 % Get all the DataFiles for all these studies
1701 for i = 1:length(sStudies)
1702 nData = length(sStudies(i).Data);
1703 argout1 = [argout1, repmat(iStudies(i), [1,nData])];
1704 argout2 = [argout2, 1:nData];
1705 end
1706
1707
1708 %% ==== DATA FOR STUDIES (INCLUDING SHARED STUDIES) ====
1709 % Usage: [iStudies, iDatas] = bst_get('DataForStudies', iStudies)
1710 case 'DataForStudies'
1711 iStudies = varargin{2};
1712 for i = 1:length(iStudies)
1713 [tmp_iStudies, tmp_iDatas] = bst_get('DataForStudy', iStudies(i));
1714 argout1 = [argout1, tmp_iStudies];
1715 argout2 = [argout2, tmp_iDatas];
1716 end
1717
1718 %% ==== DATA FILE FOR CHANNEL FILE ====
1719 % Usage: DataFiles = bst_get('DataForChannelFile', ChannelFile)
1720 case 'DataForChannelFile'
1721 ChannelFile = varargin{2};
1722 DataFiles = {};
1723 % Get study for the given channel file
1724 [sStudy, iStudy] = bst_get('ChannelFile', ChannelFile);
1725 if isempty(sStudy)
1726 return;
1727 end
1728 % Get dependent data files
1729 [iStudies, iDatas] = bst_get('DataForStudy', iStudy);
1730 % Get all the Data filenames
1731 for i = 1:length(iStudies)
1732 sStudy = bst_get('Study', iStudies(i));
1733 DataFiles = cat(2, DataFiles, {sStudy.Data(iDatas(i)).FileName});
1734 end
1735 argout1 = DataFiles;
1736
1737
1738 %% ==== RESULTS FILE ====
1739 % Usage: [sStudy, iStudy, iResult] = bst_get('ResultsFile', ResultsFile, iStudies)
1740 % [sStudy, iStudy, iResult] = bst_get('ResultsFile', ResultsFile)
1741 case 'ResultsFile'
1742 % No protocol in database
1743 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
1744 return;
1745 end
1746 % Input #2: ResultsFile
1747 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
1748 ResultsFile = varargin{2};
1749 ResultsFile = strrep(ResultsFile, ProtocolInfo.STUDIES, '');
1750 % Input #3: iStudies
1751 if (nargin < 3)
1752 iStudies = [];
1753 else
1754 iStudies = varargin{3};
1755 end
1756 % Look for surface file in all the surfaces of all subjects
1757 [argout1, argout2, argout3] = findFileInStudies('Result', 'FileName', ResultsFile, iStudies);
1758
1759
1760 %% ==== RESULTS FOR DATA FILE ====
1761 % Usage: [sStudy, iStudy, iResults] = bst_get('ResultsForDataFile', DataFile) : search the whole protocol
1762 % Usage: [sStudy, iStudy, iResults] = bst_get('ResultsForDataFile', DataFile, iStudies) : search only the specified studies
1763 case 'ResultsForDataFile'
1764 % No protocol in database
1765 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
1766 return;
1767 end
1768 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
1769 % Input #2: DataFile
1770 DataFile = varargin{2};
1771 DataFile = strrep(DataFile, ProtocolInfo.STUDIES, '');
1772 % Determine in which studies to search for ResultsFile
1773 if (nargin >= 3)
1774 % Studies specified in argument
1775 iStudy = varargin{3};
1776 else
1777 % Get study in which DataFile is located
1778 [sStudy, iStudy] = bst_get('DataFile', DataFile);
1779 if isempty(iStudy)
1780 return;
1781 end
1782 end
1783 % Search selected studies
1784 [argout1, argout2, argout3] = findFileInStudies('Result', 'DataFile', DataFile, iStudy);
1785
1786
1787 %% ==== STAT FILE ====
1788 % Usage: [sStudy, iStudy, iData] = bst_get('StatFile', StatFile, iStudies)
1789 % [sStudy, iStudy, iData] = bst_get('StatFile', StatFile)
1790 case 'StatFile'
1791 % No protocol in database
1792 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
1793 return;
1794 end
1795 % Input #2: SataFile
1796 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
1797 StatFile = varargin{2};
1798 StatFile = strrep(StatFile, ProtocolInfo.STUDIES, '');
1799 % Input #3: iStudies
1800 if (nargin < 3)
1801 iStudies = [];
1802 else
1803 iStudies = varargin{3};
1804 end
1805 % Look for surface file in all the surfaces of all subjects
1806 [argout1, argout2, argout3] = findFileInStudies('Stat', 'FileName', StatFile, iStudies);
1807
1808
1809 %% ==== STAT FOR DATA FILE ====
1810 % Usage: [sStudy, iStudy, iResults] = bst_get('StatForDataFile', DataFile) : search the whole protocol
1811 % Usage: [sStudy, iStudy, iResults] = bst_get('StatForDataFile', DataFile, iStudies) : search only the specified studies
1812 case 'StatForDataFile'
1813 % No protocol in database
1814 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
1815 return;
1816 end
1817 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
1818 % Parse inputs
1819 if (nargin >= 2)
1820 DataFile = varargin{2};
1821 DataFile = strrep(DataFile, ProtocolInfo.STUDIES, '');
1822 else
1823 error('Invalid call to bst_get().');
1824 end
1825 % Determine in which studies to search for ResultsFile
1826 if (nargin >= 3)
1827 % Studies specified in argument
1828 iStudies = varargin{3};
1829 else
1830 % Get study in which DataFile is located
1831 [sStudies, iStudies] = bst_get('DataFile', DataFile);
1832 if isempty(iStudies)
1833 return;
1834 end
1835 end
1836 % Search selected studies
1837 [argout1, argout2, argout3] = findFileInStudies('Stat', 'DataFile', DataFile, iStudies);
1838
1839 %% ==== TIMEFREQ FILE ====
1840 % Usage: [sStudy, iStudy, iTimefreq] = bst_get('TimefreqFile', TimefreqFile, iStudies)
1841 % [sStudy, iStudy, iTimefreq] = bst_get('TimefreqFile', TimefreqFile)
1842 case 'TimefreqFile'
1843 % No protocol in database
1844 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
1845 return;
1846 end
1847 % Input #2: TimefreqFile
1848 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
1849 TimefreqFile = varargin{2};
1850 TimefreqFile = strrep(TimefreqFile, ProtocolInfo.STUDIES, '');
1851 % Remove optional RefRowName
1852 iPipe = find(TimefreqFile == '|', 1);
1853 if ~isempty(iPipe)
1854 TimefreqFile = TimefreqFile(1:iPipe-1);
1855 end
1856 % Input #3: iStudies
1857 if (nargin < 3)
1858 iStudies = [];
1859 else
1860 iStudies = varargin{3};
1861 end
1862 % Look for surface file in all the surfaces of all subjects
1863 [argout1, argout2, argout3] = findFileInStudies('Timefreq', 'FileName', TimefreqFile, iStudies);
1864
1865 %% ==== TIMEFREQ FOR FILE ====
1866 % Usage: [sStudy, iStudy, iTimefreq] = bst_get('TimefreqForFile', FileName, iStudies) : search only the specified studies
1867 % [sStudy, iStudy, iTimefreq] = bst_get('TimefreqForFile', FileName) : search the whole protocol
1868 case 'TimefreqForFile'
1869 % No protocol in database
1870 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
1871 return;
1872 end
1873 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
1874 % Parse inputs
1875 if (nargin >= 2)
1876 FileName = varargin{2};
1877 FileName = strrep(FileName, ProtocolInfo.STUDIES, '');
1878 else
1879 error('Invalid call to bst_get().');
1880 end
1881 % Get study in which file is located
1882 if (nargin >= 3)
1883 iStudies = varargin{3};
1884 [sStudy, iStudy, iFile, DataType] = bst_get('AnyFile', FileName, iStudies);
1885 else
1886 [sStudy, iStudy, iFile, DataType] = bst_get('AnyFile', FileName);
1887 end
1888 % If file was not found
1889 if isempty(iStudy)
1890 return;
1891 end
1892 % Search direct dependent files
1893 [tmp, tmp, iTf] = findFileInStudies('Timefreq', 'DataFile', FileName, iStudy);
1894 % Data files: get all the depending results, and then all the timefreq for those results
1895 if strcmpi(DataType, 'data')
1896 [tmp, tmp, iResults] = bst_get('ResultsForDataFile', FileName, iStudy);
1897 for i = 1:length(iResults);
1898 % Search selected studies
1899 [tmp, tmp, iTfRes] = findFileInStudies('Timefreq', 'DataFile', sStudy.Result(iResults(i)).FileName, iStudy);
1900 if ~isempty(iTfRes)
1901 iTf = [iTf iTfRes];
1902 end
1903 end
1904 end
1905 % Return results
1906 if ~isempty(iTf)
1907 argout1 = sStudy;
1908 argout2 = iStudy;
1909 argout3 = iTf;
1910 end
1911
1912
1913 %% ==== DIPOLES FOR FILE ====
1914 % Usage: [sStudy, iStudy, iDipoles] = bst_get('DipolesForFile', FileName, iStudies) : search only the specified studies
1915 % [sStudy, iStudy, iDipoles] = bst_get('DipolesForFile', FileName) : search the whole protocol
1916 case 'DipolesForFile'
1917 % No protocol in database
1918 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
1919 return;
1920 end
1921 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
1922 % Parse inputs
1923 if (nargin >= 2)
1924 FileName = varargin{2};
1925 FileName = strrep(FileName, ProtocolInfo.STUDIES, '');
1926 else
1927 error('Invalid call to bst_get().');
1928 end
1929 % Get study in which file is located
1930 if (nargin >= 3)
1931 iStudies = varargin{3};
1932 [sStudy, iStudy, iFile, DataType] = bst_get('AnyFile', FileName, iStudies);
1933 else
1934 [sStudy, iStudy, iFile, DataType] = bst_get('AnyFile', FileName);
1935 end
1936 % If file was not found
1937 if isempty(iStudy)
1938 return;
1939 end
1940 % Search direct dependent files
1941 [tmp, tmp, iDip] = findFileInStudies('Dipoles', 'DataFile', FileName, iStudy);
1942 % Data files: get all the depending results, and then all the timefreq for those results
1943 if strcmpi(DataType, 'data')
1944 [tmp, tmp, iResults] = bst_get('ResultsForDataFile', FileName, iStudy);
1945 for i = 1:length(iResults);
1946 % Search selected studies
1947 [tmp, tmp, iDipRes] = findFileInStudies('Dipoles', 'DataFile', sStudy.Result(iResults(i)).FileName, iStudy);
1948 if ~isempty(iDipRes)
1949 iDip = [iDip, iDipRes];
1950 end
1951 end
1952 end
1953 % Return results
1954 if ~isempty(iDip)
1955 argout1 = sStudy;
1956 argout2 = iStudy;
1957 argout3 = iDip;
1958 end
1959
1960
1961 %% ==== TIMEFREQ FOR KERNEL ====
1962 % Find all the timefreq files dependent from links due to a given kernel
1963 % Usage: [sStudy, iStudy, iTimefreq] = bst_get('TimefreqForKernel', KernelFile)
1964 case 'TimefreqForKernel'
1965 sFoundStudy = [];
1966 iFoundStudy = [];
1967 iFoundTimefreq = [];
1968 % No protocol in database
1969 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
1970 return;
1971 end
1972 % Get study in which file is located
1973 KernelFile = varargin{2};
1974 [sStudy, iStudy, iFile, DataType] = bst_get('ResultsFile', KernelFile);
1975 if isempty(iStudy)
1976 return;
1977 end
1978 % Get all the data files relative with this kernel
1979 [iDepStudies, iDepDatas] = bst_get('DataForStudy', iStudy);
1980 % Keep only once each study
1981 iDepStudies = unique(iDepStudies);
1982 % Process all the dependent studies
1983 for iSt = 1:length(iDepStudies)
1984 % Get the study structure
1985 sDepStudy = bst_get('Study', iDepStudies(iSt));
1986 % Process each timefreq file separately
1987 for iTf = 1:length(sDepStudy.Timefreq)
1988 DataFile = sDepStudy.Timefreq(iTf).DataFile;
1989 % Keep only the files that are linked to LINKS
1990 if isempty(DataFile) || (length(DataFile) < 5) || ~isequal(DataFile(1:5), 'link|')
1991 continue;
1992 end
1993 % Split link
1994 splitFile = str_split(DataFile, '|');
1995 % If the kernel is found: add it to the found list
1996 if file_compare(splitFile{2}, KernelFile)
1997 sFoundStudy = [sFoundStudy sDepStudy];
1998 iFoundStudy = [iFoundStudy, iDepStudies(iSt)];
1999 iFoundTimefreq = [iFoundTimefreq, iTf];
2000 end
2001 end
2002 end
2003 % Return findings
2004 argout1 = sFoundStudy;
2005 argout2 = iFoundStudy;
2006 argout3 = iFoundTimefreq;
2007
2008
2009 %% ==== DIPOLES FOR KERNEL ====
2010 % Find all the dipoles files dependent from links due to a given kernel
2011 % Usage: [sStudy, iStudy, iDipoles] = bst_get('TimefreqForKernel', KernelFile)
2012 case 'DipolesForKernel'
2013 sFoundStudy = [];
2014 iFoundStudy = [];
2015 iFoundDipoles = [];
2016 % No protocol in database
2017 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
2018 return;
2019 end
2020 % Get study in which file is located
2021 KernelFile = varargin{2};
2022 [sStudy, iStudy, iFile, DataType] = bst_get('ResultsFile', KernelFile);
2023 if isempty(iStudy)
2024 return;
2025 end
2026 % Get all the data files relative with this kernel
2027 [iDepStudies, iDepDatas] = bst_get('DataForStudy', iStudy);
2028 % Keep only once each study
2029 iDepStudies = unique(iDepStudies);
2030 % Process all the dependent studies
2031 for iSt = 1:length(iDepStudies)
2032 % Get the study structure
2033 sDepStudy = bst_get('Study', iDepStudies(iSt));
2034 % Process each timefreq file separately
2035 for iDip = 1:length(sDepStudy.Dipoles)
2036 DataFile = sDepStudy.Dipoles(iDip).DataFile;
2037 % Keep only the files that are linked to LINKS
2038 if isempty(DataFile) || (length(DataFile) < 5) || ~isequal(DataFile(1:5), 'link|')
2039 continue;
2040 end
2041 % Split link
2042 splitFile = str_split(DataFile, '|');
2043 % If the kernel is found: add it to the found list
2044 if file_compare(splitFile{2}, KernelFile)
2045 sFoundStudy = [sFoundStudy sDepStudy];
2046 iFoundStudy = [iFoundStudy, iDepStudies(iSt)];
2047 iFoundDipoles = [iFoundDipoles, iDip];
2048 end
2049 end
2050 end
2051 % Return findings
2052 argout1 = sFoundStudy;
2053 argout2 = iFoundStudy;
2054 argout3 = iFoundDipoles;
2055
2056
2057 %% ==== DIPOLES FILE ====
2058 % Usage: [sStudy, iStudy, iDipole] = bst_get('DipolesFile', DipolesFile, iStudies)
2059 % [sStudy, iStudy, iDipole] = bst_get('DipolesFile', DipolesFile)
2060 case 'DipolesFile'
2061 % No protocol in database
2062 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
2063 return;
2064 end
2065 % Input #2: DipolesFile
2066 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
2067 DipolesFile = varargin{2};
2068 DipolesFile = strrep(DipolesFile, ProtocolInfo.STUDIES, '');
2069 % Input #3: iStudies
2070 if (nargin < 3)
2071 iStudies = [];
2072 else
2073 iStudies = varargin{3};
2074 end
2075 % Look for surface file in all the surfaces of all subjects
2076 [argout1, argout2, argout3] = findFileInStudies('Dipoles', 'FileName', DipolesFile, iStudies);
2077
2078
2079 %% ==== MATRIX FILE ====
2080 % Usage: [sStudy, iStudy, iDipole] = bst_get('MatrixFile', MatrixFile, iStudies)
2081 % [sStudy, iStudy, iDipole] = bst_get('MatrixFile', MatrixFile)
2082 case 'MatrixFile'
2083 % No protocol in database
2084 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
2085 return;
2086 end
2087 % Input #2: MatrixFile
2088 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
2089 MatrixFile = varargin{2};
2090 MatrixFile = strrep(MatrixFile, ProtocolInfo.STUDIES, '');
2091 % Input #3: iStudies
2092 if (nargin < 3)
2093 iStudies = [];
2094 else
2095 iStudies = varargin{3};
2096 end
2097 % Look for surface file in all the surfaces of all subjects
2098 [argout1, argout2, argout3] = findFileInStudies('Matrix', 'FileName', MatrixFile, iStudies);
2099
2100 %% ==== IMAGE FILE ====
2101 % Usage: [sStudy, iStudy, iDipole] = bst_get('ImageFile', ImageFile, iStudies)
2102 % [sStudy, iStudy, iDipole] = bst_get('ImageFile', ImageFile)
2103 case 'ImageFile'
2104 % No protocol in database
2105 if isempty(GlobalData.DataBase.iProtocol) || (GlobalData.DataBase.iProtocol == 0)
2106 return;
2107 end
2108 % Input #2: ImageFile
2109 ProtocolInfo = GlobalData.DataBase.ProtocolInfo(GlobalData.DataBase.iProtocol);
2110 ImageFile = varargin{2};
2111 ImageFile = strrep(ImageFile, ProtocolInfo.STUDIES, '');
2112 % Input #3: iStudies
2113 if (nargin < 3)
2114 iStudies = [];
2115 else
2116 iStudies = varargin{3};
2117 end
2118 % Look for surface file in all the surfaces of all subjects
2119 [argout1, argout2, argout3] = findFileInStudies('Image', 'FileName', ImageFile, iStudies);
2120
2121
2122 %% ==== ANY FILE ====
2123 % Usage: [sStudy, iStudy, iFile, DataType, sItem] = bst_get('AnyFile', FileName, iStudies)
2124 % [sStudy, iStudy, iFile, DataType, sItem] = bst_get('AnyFile', FileName)
2125 case 'AnyFile'
2126 % Input #2: FileName
2127 FileName = varargin{2};
2128 if isempty(FileName)
2129 return
2130 end
2131 % Input #3: iStudies
2132 if (nargin < 3)
2133 iStudies = [];
2134 else
2135 iStudies = varargin{3};
2136 end
2137 % Get data format
2138 fileType = file_gettype(FileName);
2139 if isempty(fileType)
2140 error('File type is not recognized.');
2141 end
2142 sItem = [];
2143 % Get information related with this file
2144 switch (fileType)
2145 % ===== FUNCTIONAL =====
2146 case 'channel'
2147 [sStudy, iStudy] = bst_get('ChannelFile', FileName);
2148 iItem = 1;
2149 if (nargout >= 5) && ~isempty(sStudy)
2150 sItem = sStudy.Channel;
2151 end
2152 case 'headmodel'
2153 [sStudy, iStudy, iItem] = bst_get('HeadModelFile', FileName);
2154 if (nargout >= 5) && ~isempty(sStudy)
2155 sItem = sStudy.HeadModel(iItem);
2156 end
2157 case 'noisecov'
2158 [sStudy, iStudy, iItem] = bst_get('NoiseCovFile', FileName);
2159 if (nargout >= 5) && ~isempty(sStudy)
2160 sItem = sStudy.NoiseCov(iItem);
2161 end
2162 case 'ndatacov'
2163 [sStudy, iStudy, iItem] = bst_get('DataCovFile', FileName);
2164 if (nargout >= 5) && ~isempty(sStudy)
2165 sItem = sStudy.NoiseCov(iItem);
2166 end
2167 case {'data', 'spike'}
2168 [sStudy, iStudy, iItem] = bst_get('DataFile', FileName, iStudies);
2169 if (nargout >= 5) && ~isempty(sStudy)
2170 sItem = sStudy.Data(iItem);
2171 end
2172 case {'results', 'link'}
2173 [sStudy, iStudy, iItem] = bst_get('ResultsFile', FileName, iStudies);
2174 if (nargout >= 5) && ~isempty(sStudy)
2175 sItem = sStudy.Result(iItem);
2176 end
2177 case {'presults', 'pdata','ptimefreq','pmatrix'}
2178 [sStudy, iStudy, iItem] = bst_get('StatFile', FileName, iStudies);
2179 if (nargout >= 5) && ~isempty(sStudy)
2180 sItem = sStudy.Stat(iItem);
2181 end
2182 case 'dipoles'
2183 [sStudy, iStudy, iItem] = bst_get('DipolesFile', FileName, iStudies);
2184 if (nargout >= 5) && ~isempty(sStudy)
2185 sItem = sStudy.Dipoles(iItem);
2186 end
2187 case 'timefreq'
2188 % Remove optional RefRowName
2189 iPipe = find(FileName == '|', 1);
2190 if ~isempty(iPipe)
2191 FileName = FileName(1:iPipe-1);
2192 end
2193 [sStudy, iStudy, iItem] = bst_get('TimefreqFile', FileName, iStudies);
2194 if (nargout >= 5) && ~isempty(sStudy)
2195 sItem = sStudy.Timefreq(iItem);
2196 end
2197 case 'matrix'
2198 [sStudy, iStudy, iItem] = bst_get('MatrixFile', FileName, iStudies);
2199 if (nargout >= 5) && ~isempty(sStudy)
2200 sItem = sStudy.Matrix(iItem);
2201 end
2202 case 'brainstormstudy'
2203 [sStudy, iStudy] = bst_get('Study', FileName);
2204 iItem = 0;
2205 if (nargout >= 5) && ~isempty(sStudy)
2206 sItem = sStudy;
2207 end
2208 case {'image', 'video', 'videolink'}
2209 [sStudy, iStudy, iItem] = bst_get('ImageFile', FileName, iStudies);
2210 if (nargout >= 5) && ~isempty(sStudy)
2211 sItem = sStudy.Image(iItem);
2212 end
2213 % ===== ANATOMY =====
2214 case {'cortex','scalp','innerskull','outerskull','tess','fibers','fem'}
2215 [sStudy, iStudy, iItem] = bst_get('SurfaceFile', FileName);
2216 if (nargout >= 5) && ~isempty(sStudy)
2217 sItem = sStudy.Surface(iItem);
2218 end
2219 case 'subjectimage'
2220 [sStudy, iStudy, iItem] = bst_get('MriFile', FileName);
2221 if (nargout >= 5) && ~isempty(sStudy)
2222 sItem = sStudy.Anatomy(iItem);
2223 end
2224 case 'brainstormsubject'
2225 [sStudy, iStudy] = bst_get('Subject', FileName);
2226 iItem = 0;
2227 if (nargout >= 5) && ~isempty(sStudy)
2228 sItem = sStudy;
2229 end
2230 otherwise
2231 error('File type is not recognized.');
2232 end
2233 argout1 = sStudy;
2234 argout2 = iStudy;
2235 argout3 = iItem;
2236 argout4 = fileType;
2237 if (nargout >= 5)
2238 argout5 = sItem;
2239 end
2240
2241
2242 %% ==== GET RELATED DATA FILE ====
2243 % Usage: DataFile = bst_get('RelatedDataFile', FileName, iStudies)
2244 % DataFile = bst_get('RelatedDataFile', FileName)
2245 case 'RelatedDataFile'
2246 % Input #2: FileName
2247 FileName = varargin{2};
2248 % Input #3: iStudies
2249 if (nargin < 3)
2250 iStudies = [];
2251 else
2252 iStudies = varargin{3};
2253 end
2254 % Get file in database
2255 [sStudy, iStudy, iFile, fileType] = bst_get('AnyFile', FileName, iStudies);
2256 % If this data file does not belong to any study
2257 if isempty(sStudy)
2258 return;
2259 end
2260 % Get associated data file
2261 switch (fileType)
2262 case 'data'
2263 RelatedDataFile = sStudy.Data(iFile).FileName;
2264 case {'pdata','presults','ptimefreq','pmatrix'}
2265 RelatedDataFile = sStudy.Stat(iFile).DataFile;
2266 case {'results', 'link'}
2267 RelatedDataFile = sStudy.Result(iFile).DataFile;
2268 case 'dipoles'
2269 RelatedDataFile = sStudy.Dipoles(iFile).DataFile;
2270 case 'timefreq'
2271 RelatedDataFile = sStudy.Timefreq(iFile).DataFile;
2272 otherwise
2273 RelatedDataFile = '';
2274 end
2275 % If related file is results: get related data file
2276 if ~isempty(RelatedDataFile)
2277 relFileType = file_gettype(RelatedDataFile);
2278 if ismember(relFileType, {'link','results'})
2279 RelatedDataFile = bst_get('RelatedDataFile', RelatedDataFile, iStudy);
2280 end
2281 end
2282 % Return file
2283 argout1 = RelatedDataFile;
2284
2285 %% ==== ALL CONDITIONS FOR ONE SUBJECT ====
2286 % Usage: [Conditions] = bst_get('ConditionsForSubject', SubjectFile)
2287 case 'ConditionsForSubject'
2288 % Parse inputs
2289 if (nargin == 2)
2290 SubjectFile = varargin{2};
2291 else
2292 error('Invalid call to bst_get().');
2293 end
2294 % Get list of studies associated with subject
2295 sStudies = bst_get('StudyWithSubject', SubjectFile);
2296 % Get Conditions for each study
2297 Conditions = {};
2298 for i = 1:length(sStudies)
2299 % Test if the condition of this study was not added previously
2300 isNewCondition = 1;
2301 for iCond = 1:length(Conditions)
2302 % If new condition is found
2303 % (and excludes DirAnalysisIntra and DirDefaultSubject from list)
2304 if isempty(sStudies(i).Condition) || ...
2305 isequal(sStudies(i).Condition, Conditions(iCond)) || ...
2306 strcmpi(sStudies(i).Condition{1}, bst_get('DirAnalysisIntra')) || ...
2307 strcmpi(sStudies(i).Condition{1}, bst_get('DirDefaultSubject'))
2308 isNewCondition = 0;
2309 break;
2310 end
2311 end
2312 % If Condition is not added yet : add it to the list
2313 if isNewCondition && ~isempty(sStudies(i).Condition)
2314 Conditions{end+1} = sStudies(i).Condition{1};
2315 end
2316 end
2317 % Return conditions list
2318 argout1 = Conditions;
2319
2320
2321 %% ==== ANATOMY DEFAULTS ====
2322 % Returns the list of all the anatomy defaults (distributed with the software + user defined)
2323 case 'AnatomyDefaults'
2324 % Parse inputs
2325 if (nargin == 2)
2326 AnatName = varargin{2};
2327 else
2328 AnatName = [];
2329 end
2330 % Get templates from the brainstorm3 folder
2331 progDir = bst_fullfile(bst_get('BrainstormDefaultsDir'), 'anatomy');
2332 progFiles = dir(progDir);
2333 % Get templates from the user folder
2334 userDir = bst_fullfile(bst_get('UserDefaultsDir'), 'anatomy');
2335 userFiles = dir(userDir);
2336 % Combine the two lists
2337 AllProgNames = cat(2, {progFiles.name}, cellfun(@(c)cat(2,c,'.zip'), {progFiles.name}, 'UniformOutput', 0));
2338 AllFiles = cat(2, cellfun(@(c)bst_fullfile(progDir,c), setdiff({progFiles.name}, {'.','..'}), 'UniformOutput', 0), ...
2339 cellfun(@(c)bst_fullfile(userDir,c), setdiff({userFiles.name}, AllProgNames), 'UniformOutput', 0));
2340 % Initialize list of defaults
2341 sTemplates = repmat(struct('FilePath',[],'Name',[]), 0);
2342 % Find all the valid defaults (.zip files or subdirectory with a brainstormsubject.mat in it)
2343 for i = 1:length(AllFiles)
2344 % Decompose file name
2345 [fPath, fBase, fExt] = bst_fileparts(AllFiles{i});
2346 % Entry is a directory W/ a name that does not start with a '.'
2347 if isempty(fBase) || strcmpi(fBase(1),'.') || (~isempty(fExt) && ~strcmpi(fExt, '.zip'))
2348 continue;
2349 end
2350 % If it's a folder: check for a brainstormsubject file
2351 if isdir(AllFiles{i})
2352 bstFiles = dir(bst_fullfile(AllFiles{i}, 'brainstormsubject*.mat'));
2353 if (length(bstFiles) == 1)
2354 sTemplates(end+1).FilePath = AllFiles{i};
2355 sTemplates(end).Name = fBase;
2356 end
2357 % If it's a zip file
2358 elseif isequal(fExt, '.zip')
2359 sTemplates(end+1).FilePath = AllFiles{i};
2360 sTemplates(end).Name = fBase;
2361 end
2362 end
2363 % Get defaults from internet
2364 if ~ismember('icbm152', lower({sTemplates.Name}))
2365 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=ICBM152_2023b';
2366 sTemplates(end).Name = 'ICBM152';
2367 end
2368 if ~ismember('icbm152_2022', lower({sTemplates.Name}))
2369 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=ICBM152_2022';
2370 sTemplates(end).Name = 'ICBM152_2022';
2371 end
2372 if ~ismember('icbm152_2023b', lower({sTemplates.Name}))
2373 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=ICBM152_2023b';
2374 sTemplates(end).Name = 'ICBM152_2023b';
2375 end
2376 if ~ismember('icbm152_2019', lower({sTemplates.Name}))
2377 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=ICBM152_2019';
2378 sTemplates(end).Name = 'ICBM152_2019';
2379 end
2380 if ~ismember('icbm152_brainsuite_2016', lower({sTemplates.Name}))
2381 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=ICBM152_BrainSuite_2016';
2382 sTemplates(end).Name = 'ICBM152_BrainSuite_2016';
2383 end
2384 if ~ismember('colin27_2016', lower({sTemplates.Name}))
2385 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Colin27_2016';
2386 sTemplates(end).Name = 'Colin27_2016';
2387 end
2388 if ~ismember('colin27_4nirs_2024', lower({sTemplates.Name}))
2389 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Colin27_4NIRS_2024';
2390 sTemplates(end).Name = 'Colin27_4NIRS_2024';
2391 end
2392 if ~ismember('colin27_brainsuite_2016', lower({sTemplates.Name}))
2393 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Colin27_BrainSuite_2016';
2394 sTemplates(end).Name = 'Colin27_BrainSuite_2016';
2395 end
2396 if ~ismember('bci-dni_brainsuite_2020', lower({sTemplates.Name}))
2397 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=BCI-DNI_BrainSuite_2020';
2398 sTemplates(end).Name = 'BCI-DNI_BrainSuite_2020';
2399 end
2400 if ~ismember('uscbrain_brainsuite_2020', lower({sTemplates.Name}))
2401 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=USCBrain_BrainSuite_2020';
2402 sTemplates(end).Name = 'USCBrain_BrainSuite_2020';
2403 end
2404 if ~ismember('fsaverage_2020', lower({sTemplates.Name}))
2405 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=FSAverage_2020';
2406 sTemplates(end).Name = 'FsAverage_2020';
2407 end
2408 if ~ismember('kabdebon_7w', lower({sTemplates.Name}))
2409 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Kabdebon_7w';
2410 sTemplates(end).Name = 'Kabdebon_7w';
2411 end
2412 if ~ismember('oreilly_0.5m_2021', lower({sTemplates.Name}))
2413 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Oreilly_0.5m_2021';
2414 sTemplates(end).Name = 'Oreilly_0.5m_2021';
2415 end
2416 if ~ismember('oreilly_1m_2021', lower({sTemplates.Name}))
2417 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Oreilly_1m_2021';
2418 sTemplates(end).Name = 'Oreilly_1m_2021';
2419 end
2420 if ~ismember('oreilly_2m_2021', lower({sTemplates.Name}))
2421 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Oreilly_2m_2021';
2422 sTemplates(end).Name = 'Oreilly_2m_2021';
2423 end
2424 if ~ismember(lower({sTemplates.Name}), 'oreilly_3m_2021')
2425 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Oreilly_3m_2021';
2426 sTemplates(end).Name = 'Oreilly_3m_2021';
2427 end
2428 if ~ismember('oreilly_4.5m_2021', lower({sTemplates.Name}))
2429 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Oreilly_4.5m_2021';
2430 sTemplates(end).Name = 'Oreilly_4.5m_2021';
2431 end
2432 if ~ismember('oreilly_6m_2021', lower({sTemplates.Name}))
2433 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Oreilly_6m_2021';
2434 sTemplates(end).Name = 'Oreilly_6m_2021';
2435 end
2436 if ~ismember('oreilly_7.5m_2021', lower({sTemplates.Name}))
2437 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Oreilly_7.5m_2021';
2438 sTemplates(end).Name = 'Oreilly_7.5m_2021';
2439 end
2440 if ~ismember('oreilly_9m_2021', lower({sTemplates.Name}))
2441 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Oreilly_9m_2021';
2442 sTemplates(end).Name = 'Oreilly_9m_2021';
2443 end
2444 if ~ismember('oreilly_10.5m_2021', lower({sTemplates.Name}))
2445 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Oreilly_10.5m_2021';
2446 sTemplates(end).Name = 'Oreilly_10.5m_2021';
2447 end
2448 if ~ismember('oreilly_12m_2021', lower({sTemplates.Name}))
2449 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Oreilly_12m_2021';
2450 sTemplates(end).Name = 'Oreilly_12m_2021';
2451 end
2452 if ~ismember('oreilly_15m_2021', lower({sTemplates.Name}))
2453 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Oreilly_15m_2021';
2454 sTemplates(end).Name = 'Oreilly_15m_2021';
2455 end
2456 if ~ismember('oreilly_18m_2021', lower({sTemplates.Name}))
2457 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Oreilly_18m_2021';
2458 sTemplates(end).Name = 'Oreilly_18m_2021';
2459 end
2460 if ~ismember('oreilly_24m_2021', lower({sTemplates.Name}))
2461 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=Oreilly_24m_2021';
2462 sTemplates(end).Name = 'Oreilly_24m_2021';
2463 end
2464 % If a specific template was requested
2465 if ~isempty(AnatName)
2466 iAnat = find(strcmpi({sTemplates.Name}, AnatName));
2467 sTemplates = sTemplates(iAnat);
2468 end
2469 % Sort in alphabetical order
2470 if ~isempty(sTemplates)
2471 [tmp__, I] = sort_nat({sTemplates(2:end).Name});
2472 sTemplates = sTemplates([1, I+1]);
2473 end
2474 % Return defaults list
2475 argout1 = sTemplates;
2476
2477
2478 %% ==== MNI ATLASES ====
2479 % Returns the list of all the available MNI atlases
2480 case 'MniAtlasDefaults'
2481 % Get templates from the brainstorm3 folder
2482 mniDir = bst_fullfile(bst_get('UserDefaultsDir'), 'mniatlas');
2483 mniFiles = dir(bst_fullfile(mniDir, '*.nii.gz'));
2484 mniFiles = cellfun(@(c)bst_fullfile(mniDir,c), {mniFiles.name}, 'UniformOutput', 0);
2485 % Initialize list of defaults
2486 sTemplates = repmat(struct('FilePath',[],'Name',[],'Info',[]), 0);
2487 % Find all the valid defaults (.zip files or subdirectory with a brainstormsubject.mat in it)
2488 for i = 1:length(mniFiles)
2489 % Decompose file name
2490 [fPath, fBase, fExt] = bst_fileparts(mniFiles{i});
2491 % Keep only files with .nii and .nii.gz extensions
2492 if ~isempty(fBase) && (fBase(1) ~= '.') && ~isempty(fExt) && strcmpi(fExt, '.gz')
2493 sTemplates(end+1).FilePath = mniFiles{i};
2494 sTemplates(end).Name = strrep(fBase, '.nii', '');
2495 sTemplates(end).Info = '';
2496 end
2497 end
2498 % Sort in alphabetical order
2499 if ~isempty(sTemplates)
2500 [tmp__, I] = sort_nat(lower({sTemplates.Name}));
2501 sTemplates = sTemplates(I);
2502 end
2503
2504 % Get defaults from internet
2505 if ~ismember('aal1', lower({sTemplates.Name}))
2506 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=mni_AAL1';
2507 sTemplates(end).Name = 'AAL1';
2508 sTemplates(end).Info = 'https://www.gin.cnrs.fr/en/tools/aal/';
2509 end
2510 if ~ismember('aal2', lower({sTemplates.Name}))
2511 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=mni_AAL2';
2512 sTemplates(end).Name = 'AAL2';
2513 sTemplates(end).Info = 'https://www.gin.cnrs.fr/en/tools/aal/';
2514 end
2515 if ~ismember('aal3', lower({sTemplates.Name}))
2516 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=mni_AAL3';
2517 sTemplates(end).Name = 'AAL3';
2518 sTemplates(end).Info = 'https://www.gin.cnrs.fr/en/tools/aal/';
2519 end
2520 if ~ismember('aicha', lower({sTemplates.Name}))
2521 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=mni_AICHA';
2522 sTemplates(end).Name = 'AICHA';
2523 sTemplates(end).Info = 'https://www.gin.cnrs.fr/en/tools/aicha';
2524 end
2525 if ~ismember('brainnetome', lower({sTemplates.Name}))
2526 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=mni_Brainnetome';
2527 sTemplates(end).Name = 'Brainnetome';
2528 sTemplates(end).Info = 'https://atlas.brainnetome.org/';
2529 end
2530 if ~ismember('brainnetome_leaddbs', lower({sTemplates.Name}))
2531 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=mni_Brainnetome_leaddbs';
2532 sTemplates(end).Name = 'Brainnetome_leaddbs';
2533 sTemplates(end).Info = 'https://atlas.brainnetome.org/';
2534 end
2535 if ~ismember('brodmann', lower({sTemplates.Name}))
2536 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=mni_Brodmann';
2537 sTemplates(end).Name = 'Brodmann';
2538 sTemplates(end).Info = 'https://people.cas.sc.edu/rorden/mricro/lesion.html#brod';
2539 end
2540 if ~ismember('hammers83', lower({sTemplates.Name}))
2541 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=mni_Hammers';
2542 sTemplates(end).Name = 'Hammers';
2543 sTemplates(end).Info = 'https://brain-development.org/brain-atlases/adult-brain-atlases/';
2544 end
2545 if ~ismember('neuromorphometrics', lower({sTemplates.Name}))
2546 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=mni_Neuromorphometrics';
2547 sTemplates(end).Name = 'Neuromorphometrics';
2548 sTemplates(end).Info = 'https://search.kg.ebrains.eu/instances/Dataset/ef48c5e9-6b3c-4d5a-a9a9-e678fe10bdf6';
2549 end
2550 if ~ismember('julich-brain-v25', lower({sTemplates.Name}))
2551 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=mni_Julich-Brain-v25';
2552 sTemplates(end).Name = 'Julich-Brain-v25';
2553 sTemplates(end).Info = 'https://search.kg.ebrains.eu/instances/Dataset/ef48c5e9-6b3c-4d5a-a9a9-e678fe10bdf6';
2554 end
2555 if ~ismember('schaefer2018_100_7net', lower({sTemplates.Name}))
2556 sTemplates(end+1).FilePath = 'https://neuroimage.usc.edu/bst/getupdate.php?t=mni_Schaefer2018';
2557 sTemplates(end).Name = 'Schaefer2018';
2558 sTemplates(end).Info = 'https://github.com/ThomasYeoLab/CBIG/tree/master/stable_projects/brain_parcellation/Schaefer2018_LocalGlobal';
2559 end
2560 % Return defaults list
2561 argout1 = sTemplates;
2562
2563
2564 %% ==== EEG DEFAULTS ====
2565 % Returns an array of struct(fullpath, name) of all the Brainstorm eeg nets defaults
2566 % Usage: EegDefaults = bst_get('EegDefaults')
2567 % EegDefaults = bst_get('EegDefaults', TemplateName=[], SetupName=[])
2568 case 'EegDefaults'
2569 % Parse inputs
2570 if (nargin >= 3)
2571 SetupName = varargin{3};
2572 else
2573 SetupName = [];
2574 end
2575 if (nargin >= 2)
2576 TemplateName = varargin{2};
2577 else
2578 TemplateName = [];
2579 end
2580 % Get templates from the brainstorm3 folder
2581 progDir = bst_fullfile(bst_get('BrainstormDefaultsDir'), 'eeg');
2582 progFiles = dir(bst_fullfile(progDir, '*'));
2583 % Get templates from the user folder
2584 userDir = bst_fullfile(bst_get('UserDefaultsDir'), 'eeg');
2585 userFiles = dir(bst_fullfile(userDir, '*'));
2586 % Combine the two lists
2587 dirList = cat(2, cellfun(@(c)bst_fullfile(progDir,c), {progFiles.name}, 'UniformOutput', 0), ...
2588 cellfun(@(c)bst_fullfile(userDir,c), setdiff({userFiles.name}, {progFiles.name}), 'UniformOutput', 0));
2589 % Initialize list of folders
2590 fullDefaultsList = repmat(struct('contents','', 'name',''), 0);
2591 % For each template directory
2592 for iDir = 1:length(dirList)
2593 % Decompose file name
2594 [fPath, fBase, fExt] = bst_fileparts(dirList{iDir});
2595 % Entry is a not a folder, or starts with a "."
2596 if ~isdir(dirList{iDir}) || isempty(fBase) || strcmpi(fBase(1),'.')
2597 continue;
2598 end
2599 % Skip if it is not the requested template
2600 if ~isempty(TemplateName) && ~strcmpi(fBase, TemplateName)
2601 continue;
2602 end
2603 % Get files list
2604 fileList = dir(bst_fullfile(dirList{iDir}, 'channel*.mat'));
2605 defaultsList = repmat(struct('fullpath','', 'name',''), 0);
2606 % Find all the valid defaults (channel files)
2607 for iFile = 1:length(fileList)
2608 [tmp__, baseName] = bst_fileparts(fileList(iFile).name);
2609 baseName = strrep(baseName, 'channel_', '');
2610 baseName = strrep(baseName, '_channel', '');
2611 baseName = strrep(baseName, '_', ' ');
2612 % Skip if it is not the requested template
2613 if ~isempty(SetupName) && ~strcmpi(baseName, SetupName)
2614 continue;
2615 end
2616 % Add to list of templates
2617 iNewDefault = length(defaultsList) + 1;
2618 defaultsList(iNewDefault).fullpath = bst_fullfile(dirList{iDir}, fileList(iFile).name);
2619 defaultsList(iNewDefault).name = baseName;
2620 end
2621 % Add files list to defaults list
2622 if ~isempty(defaultsList)
2623 fullDefaultsList(end + 1) = struct('contents', defaultsList, ...
2624 'name', fBase);
2625 end
2626 end
2627 % Return defaults list
2628 argout1 = fullDefaultsList;
2629
2630
2631 %% ==== GET FILENAMES ====
2632 case 'GetFilenames'
2633 iStudies = varargin{2};
2634 iItems = varargin{3};
2635 DataType = varargin{4};
2636 FileNames = cell(1, length(iStudies));
2637 argout1 = {};
2638 for i = 1:length(iStudies)
2639 % Get study definition
2640 sStudy = bst_get('Study', iStudies(i));
2641 if isempty(sStudy)
2642 continue;
2643 end
2644 % Recordings or sources
2645 switch (DataType)
2646 case 'data'
2647 if (iItems(i) > length(sStudy.Data))
2648 return;
2649 end
2650 FileNames{i} = sStudy.Data(iItems(i)).FileName;
2651 case 'results'
2652 if (iItems(i) > length(sStudy.Result))
2653 return;
2654 end
2655 FileNames{i} = sStudy.Result(iItems(i)).FileName;
2656 case 'timefreq'
2657 if (iItems(i) > length(sStudy.Timefreq))
2658 return;
2659 end
2660 FileNames{i} = sStudy.Timefreq(iItems(i)).FileName;
2661 case 'matrix'
2662 if (iItems(i) > length(sStudy.Matrix))
2663 return;
2664 end
2665 FileNames{i} = sStudy.Matrix(iItems(i)).FileName;
2666 case {'pdata','presults','ptimfreq'}
2667 if (iItems(i) > length(sStudy.Stat))
2668 return;
2669 end
2670 FileNames{i} = sStudy.Stat(iItems(i)).FileName;
2671 end
2672 end
2673 argout1 = FileNames;
2674
2675
2676 %% ==== GUI ====
2677 case 'BstFrame'
2678 if isempty(GlobalData) || isempty(GlobalData.Program.GUI) || isempty(GlobalData.Program.GUI.mainWindow)
2679 argout1 = [];
2680 else
2681 argout1 = GlobalData.Program.GUI.mainWindow.jBstFrame;
2682 end
2683 case 'BstControls'
2684 if isempty(GlobalData) || isempty(GlobalData.Program) || isempty(GlobalData.Program.GUI) || isempty(GlobalData.Program.GUI.mainWindow)
2685 argout1 = [];
2686 else
2687 argout1 = GlobalData.Program.GUI.mainWindow;
2688 end
2689 case 'isGUI'
2690 if isempty(GlobalData) || isempty(GlobalData.Program) || ~isfield(GlobalData.Program, 'GuiLevel')
2691 argout1 = [];
2692 else
2693 argout1 = (GlobalData.Program.GuiLevel >= 1);
2694 end
2695 case 'GuiLevel'
2696 if isempty(GlobalData) || isempty(GlobalData.Program) || ~isfield(GlobalData.Program, 'GuiLevel')
2697 argout1 = [];
2698 else
2699 argout1 = GlobalData.Program.GuiLevel;
2700 end
2701 case 'ScreenDef'
2702 if isempty(GlobalData) || isempty(GlobalData.Program) || ~isfield(GlobalData.Program, 'ScreenDef')
2703 argout1 = [];
2704 else
2705 argout1 = GlobalData.Program.ScreenDef;
2706 end
2707 case 'DecorationSize'
2708 if isempty(GlobalData) || isempty(GlobalData.Program) || ~isfield(GlobalData.Program, 'DecorationSize')
2709 argout1 = [];
2710 else
2711 argout1 = GlobalData.Program.DecorationSize;
2712 end
2713 case 'Layout'
2714 % Default or current layout structure
2715 if ~isfield(GlobalData, 'Preferences') || ~isfield(GlobalData.Preferences, 'Layout') || ~((nargin == 1) || isfield(GlobalData.Preferences.Layout, varargin{2})) || ~isfield(GlobalData.Preferences.Layout, 'MainWindowPos')
2716 GlobalData.Preferences.Layout = db_template('Layout');
2717 end
2718 % Structure or property call
2719 if (nargin == 2) && ischar(varargin{2}) && isfield(GlobalData.Preferences.Layout, varargin{2})
2720 argout1 = GlobalData.Preferences.Layout.(varargin{2});
2721 elseif (nargin == 1)
2722 argout1 = GlobalData.Preferences.Layout;
2723 else
2724 error('Invalid call to bst_get.');
2725 end
2726
2727 case 'ByteOrder'
2728 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'ByteOrder')
2729 argout1 = GlobalData.Preferences.ByteOrder;
2730 else
2731 argout1 = 'l';
2732 end
2733
2734 case 'UniformizeTimeSeriesScales'
2735 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'UniformizeTimeSeriesScales')
2736 argout1 = GlobalData.Preferences.UniformizeTimeSeriesScales;
2737 else
2738 argout1 = 1;
2739 end
2740
2741 case 'FlipYAxis'
2742 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'FlipYAxis')
2743 argout1 = GlobalData.Preferences.FlipYAxis;
2744 else
2745 argout1 = 0;
2746 end
2747
2748 case 'AutoScaleY'
2749 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'AutoScaleY')
2750 argout1 = GlobalData.Preferences.AutoScaleY;
2751 else
2752 argout1 = 1;
2753 end
2754
2755 case 'ShowXGrid'
2756 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'ShowXGrid')
2757 argout1 = GlobalData.Preferences.ShowXGrid;
2758 else
2759 argout1 = 0;
2760 end
2761
2762 case 'ShowYGrid'
2763 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'ShowYGrid')
2764 argout1 = GlobalData.Preferences.ShowYGrid;
2765 else
2766 argout1 = 0;
2767 end
2768
2769 case 'ShowZeroLines'
2770 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'ShowZeroLines')
2771 argout1 = GlobalData.Preferences.ShowZeroLines;
2772 else
2773 argout1 = 1;
2774 end
2775
2776 case 'Resolution'
2777 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'Resolution')
2778 argout1 = GlobalData.Preferences.Resolution;
2779 else
2780 argout1 = [0 0];
2781 end
2782
2783 case 'FixedScaleY'
2784 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'FixedScaleY') && isfield(GlobalData.Preferences.FixedScaleY, varargin{2}) && ~isempty(GlobalData.Preferences.FixedScaleY.(varargin{2}))
2785 argout1 = GlobalData.Preferences.FixedScaleY.(varargin{2});
2786 else
2787 argout1 = [];
2788 end
2789
2790 case 'XScale'
2791 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'XScale')
2792 argout1 = GlobalData.Preferences.XScale;
2793 else
2794 argout1 = 'linear';
2795 end
2796
2797 case 'YScale'
2798 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'YScale')
2799 argout1 = GlobalData.Preferences.YScale;
2800 else
2801 argout1 = 'linear';
2802 end
2803
2804 case 'ShowEventsMode'
2805 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'ShowEventsMode')
2806 argout1 = GlobalData.Preferences.ShowEventsMode;
2807 else
2808 argout1 = 'dot';
2809 end
2810
2811 case 'AutoUpdates'
2812 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'AutoUpdates')
2813 argout1 = GlobalData.Preferences.AutoUpdates;
2814 else
2815 argout1 = 1;
2816 end
2817
2818 case 'ForceMatCompression'
2819 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'ForceMatCompression')
2820 argout1 = GlobalData.Preferences.ForceMatCompression;
2821 else
2822 argout1 = 0;
2823 end
2824
2825 case 'IgnoreMemoryWarnings'
2826 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'IgnoreMemoryWarnings')
2827 argout1 = GlobalData.Preferences.IgnoreMemoryWarnings;
2828 else
2829 argout1 = 0;
2830 end
2831
2832 case 'SystemCopy'
2833 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'SystemCopy')
2834 argout1 = GlobalData.Preferences.SystemCopy;
2835 else
2836 argout1 = 0;
2837 end
2838
2839 case 'ExpertMode'
2840 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'ExpertMode')
2841 argout1 = GlobalData.Preferences.ExpertMode;
2842 else
2843 argout1 = 0;
2844 end
2845
2846 case 'DisplayGFP'
2847 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'DisplayGFP')
2848 argout1 = GlobalData.Preferences.DisplayGFP;
2849 else
2850 argout1 = 1;
2851 end
2852
2853 case 'DownsampleTimeSeries'
2854 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'DownsampleTimeSeries')
2855 if (GlobalData.Preferences.DownsampleTimeSeries == 1)
2856 GlobalData.Preferences.DownsampleTimeSeries = 5;
2857 end
2858 argout1 = GlobalData.Preferences.DownsampleTimeSeries;
2859 else
2860 argout1 = 5;
2861 end
2862
2863 case 'GraphicsSmoothing'
2864 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'GraphicsSmoothing')
2865 argout1 = GlobalData.Preferences.GraphicsSmoothing;
2866 else
2867 argout1 = 5;
2868 end
2869
2870 case 'UseCrossPlatformJLF'
2871 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'UseCrossPlatformJLF')
2872 argout1 = GlobalData.Preferences.UseCrossPlatformJLF;
2873 else
2874 argout1 = 0;
2875 end
2876
2877 case 'ShowProcessTooltip'
2878 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'ShowProcessTooltip')
2879 argout1 = GlobalData.Preferences.ShowProcessTooltip;
2880 else
2881 argout1 = 0;
2882 end
2883
2884 case 'DisableOpenGL'
2885 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'DisableOpenGL')
2886 argout1 = GlobalData.Preferences.DisableOpenGL;
2887 else
2888 argout1 = 0;
2889 end
2890
2891 case 'InterfaceScaling'
2892 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'InterfaceScaling')
2893 argout1 = GlobalData.Preferences.InterfaceScaling;
2894 else
2895 % Get screen resolution
2896 if isfield(GlobalData, 'Program') && isfield(GlobalData.Program, 'ScreenDef') && isfield(GlobalData.Program.ScreenDef, 'javaPos') && ~isempty(GlobalData.Program.ScreenDef(1).javaPos)
2897 AvailableRes = [100 125 150 200 250 300 400];
2898 iRes = bst_closest(GlobalData.Program.ScreenDef(1).javaPos.width * 100 / 1920, AvailableRes);
2899 argout1 = AvailableRes(iRes);
2900 else
2901 argout1 = 100;
2902 end
2903 end
2904
2905 case 'TSDisplayMode'
2906 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'TSDisplayMode')
2907 argout1 = GlobalData.Preferences.TSDisplayMode;
2908 else
2909 argout1 = 'butterfly';
2910 end
2911
2912 case 'PluginCustomPath'
2913 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'PluginCustomPath') && ~isempty(GlobalData.Preferences.PluginCustomPath)
2914 argout1 = GlobalData.Preferences.PluginCustomPath;
2915 else
2916 argout1 = [];
2917 end
2918
2919 case 'BrainSuiteDir'
2920 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'BrainSuiteDir') && ~isempty(GlobalData.Preferences.BrainSuiteDir)
2921 if isdir(GlobalData.Preferences.BrainSuiteDir) && file_exist(bst_fullfile(GlobalData.Preferences.BrainSuiteDir, 'bdp'))
2922 argout1 = GlobalData.Preferences.BrainSuiteDir;
2923 else
2924 argout1 = [];
2925 end
2926 else
2927 argout1 = [];
2928 end
2929
2930 case 'SpmTpmAtlas'
2931 preferSpm = 0;
2932 % CALL: bst_get('SpmTpmAtlas', 'SPM')
2933 if (nargin >= 2) && strcmpi(varargin{2}, 'SPM')
2934 preferSpm = 1;
2935 end
2936
2937 % Get template file
2938 tpmUser = bst_fullfile(bst_get('BrainstormUserDir'), 'defaults', 'spm', 'TPM.nii');
2939 if file_exist(tpmUser) && ~preferSpm
2940 argout1 = tpmUser;
2941 disp(['BST> SPM12 template found: ' tpmUser]);
2942 return;
2943 end
2944 % If it does not exist: check in brainstorm3 folder
2945 tpmDistrib = bst_fullfile(bst_get('BrainstormHomeDir'), 'defaults', 'spm', 'TPM.nii');
2946 if file_exist(tpmDistrib) && ~preferSpm
2947 argout1 = tpmDistrib;
2948 disp(['BST> SPM12 template found: ' tpmDistrib]);
2949 return;
2950 end
2951 % If it does not exist: check in spm12 folder
2952 PlugSpm = bst_plugin('GetInstalled', 'spm12');
2953 if ~isempty(PlugSpm)
2954 tpmSpm = bst_fullfile(PlugSpm.Path, PlugSpm.SubFolder, 'tpm', 'TPM.nii');
2955 if file_exist(tpmSpm)
2956 argout1 = tpmSpm;
2957 disp(['BST> SPM12 template found: ' tpmSpm]);
2958 return;
2959 elseif preferSpm
2960 argout1 = bst_get('SpmTpmAtlas');
2961 return
2962 end
2963 else
2964 tpmSpm = '';
2965 end
2966 % Not found...
2967 disp('SPM12 template not found in any of the following folders:');
2968 disp([' - ' tpmUser]);
2969 disp([' - ' tpmDistrib]);
2970 if ~isempty(tpmSpm)
2971 disp([' - ' tpmSpm]);
2972 end
2973 argout1 = [];
2974
2975 case 'PythonExe'
2976 % Get saved value
2977 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'PythonExe') && ~isempty(GlobalData.Preferences.PythonExe)
2978 if file_exist(GlobalData.Preferences.PythonExe)
2979 argout1 = GlobalData.Preferences.PythonExe;
2980 else
2981 disp(['BST> Error: Python executable not found: ' GlobalData.Preferences.PythonExe]);
2982 argout1 = [];
2983 end
2984 else
2985 argout1 = [];
2986 end
2987 % If not defined in Brainstorm, but set in Matlab
2988 if isempty(argout1)
2989 [pyVer, PythonExe] = bst_python_ver();
2990 if ~isempty(PythonExe) && file_exist(PythonExe)
2991 disp(['BST> Found Python executable: ' PythonExe]);
2992 argout1 = PythonExe;
2993 bst_set('PythonExe', PythonExe);
2994 end
2995 end
2996
2997 case 'ElectrodeConfig'
2998 % Get modality
2999 Modality = varargin{2};
3000 if isequal(Modality, 'ECOG+SEEG')
3001 Modality = 'ECOG_SEEG';
3002 elseif isempty(Modality) || ~ismember(Modality, {'EEG','ECOG','SEEG'})
3003 error(['Invalid modality: ' Modality]);
3004 end
3005 % Value was saved previously
3006 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'ElectrodeConfig') && isfield(GlobalData.Preferences.ElectrodeConfig, Modality) && isfield(GlobalData.Preferences.ElectrodeConfig.(Modality), 'ContactDiameter')
3007 argout1 = GlobalData.Preferences.ElectrodeConfig.(Modality);
3008 % Get default value
3009 else
3010 switch (Modality)
3011 case 'EEG'
3012 ElectrodeConfig.Type = 'eeg';
3013 ElectrodeConfig.ContactDiameter = 0.010;
3014 ElectrodeConfig.ContactLength = 0.002;
3015 ElectrodeConfig.ElecDiameter = [];
3016 ElectrodeConfig.ElecLength = [];
3017 case 'ECOG'
3018 ElectrodeConfig.Type = 'ecog';
3019 ElectrodeConfig.ContactDiameter = 0.004;
3020 ElectrodeConfig.ContactLength = 0.001;
3021 ElectrodeConfig.ElecDiameter = 0.0005;
3022 ElectrodeConfig.ElecLength = [];
3023 case {'SEEG','ECOG_SEEG'}
3024 ElectrodeConfig.Type = 'seeg';
3025 ElectrodeConfig.ContactDiameter = 0.0008;
3026 ElectrodeConfig.ContactLength = 0.002;
3027 ElectrodeConfig.ElecDiameter = 0.0007;
3028 ElectrodeConfig.ElecLength = 0.070;
3029 end
3030 argout1 = ElectrodeConfig;
3031 end
3032
3033 case 'ElecInterpDist'
3034 % Get modality
3035 Modality = varargin{2};
3036 if isequal(Modality, 'ECOG+SEEG')
3037 Modality = 'ECOG_SEEG';
3038 elseif isempty(Modality) || ~ismember(Modality, {'EEG','ECOG','SEEG','MEG'})
3039 error(['Invalid modality: ' Modality]);
3040 end
3041 % Value was saved previously
3042 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'ElecInterpDist') && isfield(GlobalData.Preferences.ElecInterpDist, Modality)
3043 argout1 = GlobalData.Preferences.ElecInterpDist.(Modality);
3044 % Get default value
3045 else
3046 switch (Modality)
3047 case 'EEG', argout1 = .3;
3048 case 'ECOG', argout1 = .015;
3049 case 'SEEG', argout1 = .015;
3050 case 'ECOG_SEEG', argout1 = .015;
3051 case 'MEG', argout1 = .5;
3052 end
3053 end
3054
3055 case 'UseSigProcToolbox'
3056 % In a parfor loop: GlobalData is empty => Check only if the toolbox is installed (ignore user preferences)
3057 if isempty(GlobalData) || ~isfield(GlobalData, 'Program') || ~isfield(GlobalData.Program, 'HasSigProcToolbox')
3058 argout1 = exist('kaiserord', 'file');
3059 else
3060 % Save the result of the check for the SigProc tb
3061 if isempty(GlobalData.Program.HasSigProcToolbox)
3062 % Check if Signal Processing Toolbox is installed
3063 GlobalData.Program.HasSigProcToolbox = (exist('kaiserord', 'file') == 2);
3064 end
3065 % Return user preferences
3066 if ~GlobalData.Program.HasSigProcToolbox
3067 argout1 = 0;
3068 elseif isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'UseSigProcToolbox')
3069 argout1 = GlobalData.Preferences.UseSigProcToolbox;
3070 else
3071 argout1 = 1;
3072 end
3073 end
3074
3075 case 'CustomColormaps'
3076 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'CustomColormaps') && ~isempty(GlobalData.Preferences.CustomColormaps)
3077 argout1 = GlobalData.Preferences.CustomColormaps;
3078 else
3079 argout1 = repmat(struct('Name', '', 'CMap', []), 0);
3080 end
3081
3082 case 'BFSProperties'
3083 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'BFSProperties') && ~isempty(GlobalData.Preferences.BFSProperties)
3084 argout1 = GlobalData.Preferences.BFSProperties;
3085 else
3086 argout1 = [.33 .0042 .33 .88 .93];
3087 end
3088
3089 case 'ShowHiddenFiles'
3090 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'ShowHiddenFiles')
3091 argout1 = GlobalData.Preferences.ShowHiddenFiles;
3092 else
3093 argout1 = 0;
3094 end
3095
3096 case 'LastUsedDirs'
3097 defPref = struct(...
3098 'ImportData', '', ...
3099 'ImportChannel', '', ...
3100 'ImportAnat', '', ...
3101 'ImportMontage', '', ...
3102 'ExportChannel', '', ...
3103 'ExportData', '', ...
3104 'ExportAnat', '', ...
3105 'ExportProtocol', '', ...
3106 'ExportImage', '', ...
3107 'ExportScript', '', ...
3108 'ExportMontage', '');
3109 argout1 = FillMissingFields(contextName, defPref);
3110 % Check that all folders are valid
3111 fields = fieldnames(argout1);
3112 for i = 1:length(fields)
3113 if ~ischar(argout1.(fields{i})) || ~file_exist(argout1.(fields{i}))
3114 argout1.(fields{i}) = '';
3115 end
3116 end
3117
3118 case 'DefaultFormats'
3119 defPref = struct(...
3120 'AnatIn', 'FreeSurfer', ...
3121 'ChannelIn', '', ...
3122 'ChannelOut', '', ...
3123 'DataIn', 'CTF', ...
3124 'DataOut', '', ...
3125 'DipolesIn', '', ...
3126 'DipolesOut', '', ...
3127 'ImageOut', '', ...
3128 'EventsIn', '', ...
3129 'EventsOut', '', ...
3130 'MriIn', '', ...
3131 'MriOut', 'Nifti1', ...
3132 'NoiseCovIn', '', ...
3133 'NoiseCovOut', '', ...
3134 'ResultsIn', '', ...
3135 'ResultsOut', '', ...
3136 'SpmOut', '', ...
3137 'SspIn', '', ...
3138 'SspOut', '', ...
3139 'SurfaceIn', '', ...
3140 'SurfaceOut', '', ...
3141 'LabelIn', '', ...
3142 'LabelOut', '', ...
3143 'TimefreqIn', '', ...
3144 'TimefreqOut', '', ...
3145 'MatrixIn', '', ...
3146 'MatrixOut', '', ...
3147 'MontageIn', '', ...
3148 'MontageOut', '', ...
3149 'FibersIn', '');
3150 argout1 = FillMissingFields(contextName, defPref);
3151
3152 case 'ImportDataOptions'
3153 defPref = db_template('ImportOptions');
3154 argout1 = FillMissingFields(contextName, defPref);
3155
3156 case 'RawViewerOptions'
3157 defPref = struct(...
3158 'PageDuration', 3, ...
3159 'RemoveBaseline', 'all', ...
3160 'UseCtfComp', 1, ...
3161 'Shortcuts', []);
3162 defPref.Shortcuts = {...
3163 '1', 'event1', 'simple', []; ... % Key, event name, event type (simple,extended,page), epoch time
3164 '2', 'event2', 'simple', []; ...
3165 '3', 'event3', 'simple', []; ...
3166 '4', 'event4', 'simple', []; ...
3167 '5', 'event5', 'simple', []; ...
3168 '6', 'event6', 'simple', []; ...
3169 '7', 'event7', 'simple', []; ...
3170 '8', 'event8', 'simple', []; ...
3171 '9', 'event9', 'simple', []};
3172 argout1 = FillMissingFields(contextName, defPref);
3173 % If invalid PageDuration: reset to default
3174 if (argout1.PageDuration <= 0.1)
3175 argout1.PageDuration = defPref.PageDuration;
3176 end
3177 % If old shortcuts: reset to defaults
3178 if any(size(argout1.Shortcuts) ~= size(defPref.Shortcuts))
3179 disp('BST> Warning: Reset keyboard shortcuts to include new options.');
3180 argout1.Shortcuts = defPref.Shortcuts;
3181 bst_set('RawViewerOptions', argout1);
3182 end
3183
3184 case 'MontageOptions'
3185 defPref = struct('Shortcuts', []);
3186 defPref.Shortcuts = {
3187 %'a', []; ... Note: A is reserved for All channels
3188 'b', []; ...
3189 'c', []; ...
3190 'd', []; ...
3191 'e', []; ...
3192 'f', []; ...
3193 'g', []; ...
3194 'h', []; ...
3195 'i', []; ...
3196 'j', []; ...
3197 'k', []; ...
3198 'l', []; ...
3199 'm', []; ...
3200 'n', []; ...
3201 'o', []; ...
3202 'p', []; ...
3203 'q', []; ...
3204 'r', []; ...
3205 's', []; ...
3206 't', []; ...
3207 'u', []; ...
3208 'v', []; ...
3209 'w', []; ...
3210 'x', []; ...
3211 'y', []; ...
3212 'z', []; ...
3213 };
3214 argout1 = FillMissingFields(contextName, defPref);
3215
3216 case 'TopoLayoutOptions'
3217 defPref = struct(...
3218 'TimeWindow', [], ...
3219 'FreqWindow', [], ...
3220 'WhiteBackground', 0, ...
3221 'ShowRefLines', 1, ...
3222 'ShowLegend', 1, ...
3223 'FlipYAxis', 0, ...
3224 'ContourLines', 10, ...
3225 'ShowHeadLines', 0);
3226 argout1 = FillMissingFields(contextName, defPref);
3227
3228 case 'StatThreshOptions'
3229 defPref = struct(...
3230 'pThreshold', .05, ...
3231 'durThreshold', 0, ...
3232 'Correction', 'fdr', ...
3233 'Control', [1 2 3]);
3234 argout1 = FillMissingFields(contextName, defPref);
3235 % Make sure that Control is not a string (previous brainstorm version)
3236 if ischar(argout1.Control)
3237 argout1.Control = defPref.Control;
3238 end
3239 % Make sure that 'no' is used instead of 'none' (previous brainstorm version)
3240 if strcmpi(argout1.Correction, 'none')
3241 argout1.Correction = 'no';
3242 end
3243
3244 case 'ContactSheetOptions'
3245 defPref = struct(...
3246 'nImages', 20, ...
3247 'TimeRange', [], ...
3248 'SkipVolume', 0.2);
3249 argout1 = FillMissingFields(contextName, defPref);
3250
3251 case 'ProcessOptions'
3252 defPref = struct(...
3253 'SavedParam', struct(), ...
3254 'MaxBlockSize', 100 * 1024 * 1024 / 8, ... % 100MiB == 13,107,200 doubles
3255 'LastMaxBlockSize', 100 * 1024 * 1024 / 8); % 100MiB == 13,107,200 doubles
3256 argout1 = FillMissingFields(contextName, defPref);
3257
3258 case 'ImportEegRawOptions'
3259 defPref = struct(...
3260 'isCanceled', 0, ...
3261 'BaselineDuration', 0, ...
3262 'SamplingRate', 1000, ...
3263 'MatrixOrientation', 'channelXtime', ... % {'channelXtime', 'timeXchannel'}
3264 'VoltageUnits', 'V', ... % {'\muV', 'mV', 'V'}
3265 'SkipLines', 0, ...
3266 'nAvg', 1, ...
3267 'isChannelName', 0); % 1 if the first entry contains the channel name
3268 argout1 = FillMissingFields(contextName, defPref);
3269
3270 case 'BugReportOptions'
3271 defPref = struct(...
3272 'isEnabled', 0, ...
3273 'SmtpServer', 'mailhost.chups.jussieu.fr', ...
3274 'UserEmail', '');
3275 argout1 = FillMissingFields(contextName, defPref);
3276
3277 case 'DefaultSurfaceDisplay'
3278 defPref = struct(...
3279 'SurfShowSulci', 1, ...
3280 'SurfSmoothValue', 0, ...
3281 'DataThreshold', 0.5, ...
3282 'SizeThreshold', 1, ...
3283 'DataAlpha', 0);
3284 argout1 = FillMissingFields(contextName, defPref);
3285
3286 case 'MagneticExtrapOptions'
3287 defPref = struct(...
3288 'ForceWhitening', 0, ...
3289 'EpsilonValue', 0.0001);
3290 argout1 = FillMissingFields(contextName, defPref);
3291
3292 case 'DefaultFreqBands'
3293 argout1 = {...
3294 'delta', '2, 4', 'mean'; ...
3295 'theta', '5, 7', 'mean'; ...
3296 'alpha', '8, 12', 'mean'; ...
3297 'beta', '15, 29', 'mean'; ...
3298 'gamma1', '30, 59', 'mean'; ...
3299 'gamma2', '60, 90', 'mean'};
3300
3301 case 'TimefreqOptions_morlet'
3302 defPref.isTimeBands = 0;
3303 defPref.isFreqBands = 0;
3304 defPref.isFreqLog = 0;
3305 defPref.TimeBands = {};
3306 defPref.Freqs = '1:1:60';
3307 defPref.FreqsLog = '1:40:150';
3308 defPref.FreqBands = bst_get('DefaultFreqBands');
3309 defPref.Measure = 'power';
3310 defPref.SaveKernel = 0;
3311 defPref.Output = 'all';
3312 defPref.RemoveEvoked = 0;
3313 defPref.ClusterFuncTime = 'after';
3314 defPref.MorletFc = 1;
3315 defPref.MorletFwhmTc = 3;
3316 argout1 = FillMissingFields(contextName, defPref);
3317 if isempty(argout1.Freqs)
3318 argout1.Freqs = defPref.Freqs;
3319 end
3320 if ~isempty(argout1.FreqBands) && ((size(argout1.FreqBands,2) ~= 3) || ~all(cellfun(@ischar, argout1.FreqBands(:))) || any(cellfun(@(c)isempty(strtrim(c)), argout1.FreqBands(:))))
3321 argout1.FreqBands = defPref.FreqBands;
3322 end
3323
3324 case 'TimefreqOptions_hilbert'
3325 defPref.isTimeBands = 0;
3326 defPref.isFreqBands = 1;
3327 defPref.isFreqLog = 0;
3328 defPref.TimeBands = {};
3329 defPref.Freqs = [];
3330 defPref.FreqsLog = [];
3331 defPref.FreqBands = bst_get('DefaultFreqBands');
3332 defPref.Measure = 'power';
3333 defPref.SaveKernel = 0;
3334 defPref.Output = 'all';
3335 defPref.RemoveEvoked = 0;
3336 defPref.ClusterFuncTime = 'after';
3337 argout1 = FillMissingFields(contextName, defPref);
3338 if isempty(argout1.Freqs)
3339 argout1.Freqs = defPref.Freqs;
3340 end
3341 if ~isempty(argout1.FreqBands) && (size(argout1.FreqBands,2) == 3) && ~ischar(argout1.FreqBands{1,2})
3342 argout1.FreqBands = defPref.FreqBands;
3343 end
3344
3345 case 'TimefreqOptions_plv' % not used
3346 defPref.isTimeBands = 0;
3347 defPref.isFreqBands = 1;
3348 defPref.isFreqLog = 0;
3349 defPref.TimeBands = {};
3350 defPref.Freqs = [];
3351 defPref.FreqsLog = [];
3352 defPref.FreqBands = bst_get('DefaultFreqBands');
3353 defPref.Measure = 'other';
3354 defPref.SaveKernel = 0;
3355 defPref.Output = 'all';
3356 defPref.ClusterFuncTime = 'after';
3357 argout1 = FillMissingFields(contextName, defPref);
3358 if isempty(argout1.Freqs)
3359 argout1.Freqs = defPref.Freqs;
3360 end
3361 if ~isempty(argout1.FreqBands) && ~ischar(argout1.FreqBands{1,2})
3362 argout1.FreqBands = defPref.FreqBands;
3363 end
3364
3365 case 'TimefreqOptions_fft'
3366 defPref.isTimeBands = 0;
3367 defPref.isFreqBands = 0;
3368 defPref.isFreqLog = 0;
3369 defPref.TimeBands = {};
3370 defPref.Freqs = [];
3371 defPref.FreqsLog = [];
3372 defPref.FreqBands = bst_get('DefaultFreqBands');
3373 defPref.Measure = 'power';
3374 defPref.Output = 'all';
3375 defPref.ClusterFuncTime = 'after';
3376 argout1 = FillMissingFields(contextName, defPref);
3377 if isempty(argout1.Freqs)
3378 argout1.Freqs = defPref.Freqs;
3379 end
3380 if ~isempty(argout1.FreqBands) && ~ischar(argout1.FreqBands{1,2})
3381 argout1.FreqBands = defPref.FreqBands;
3382 end
3383
3384 case 'TimefreqOptions_psd'
3385 defPref.isTimeBands = 0;
3386 defPref.isFreqBands = 0;
3387 defPref.isFreqLog = 0;
3388 defPref.TimeBands = {};
3389 defPref.Freqs = [];
3390 defPref.FreqsLog = [];
3391 defPref.FreqBands = bst_get('DefaultFreqBands');
3392 defPref.Measure = 'power';
3393 defPref.Output = 'all';
3394 defPref.ClusterFuncTime = 'after';
3395 argout1 = FillMissingFields(contextName, defPref);
3396 if isempty(argout1.Freqs)
3397 argout1.Freqs = defPref.Freqs;
3398 end
3399 if ~isempty(argout1.FreqBands) && ~ischar(argout1.FreqBands{1,2})
3400 argout1.FreqBands = defPref.FreqBands;
3401 end
3402
3403 case 'TimefreqOptions_stft'
3404 defPref.isTimeBands = 0;
3405 defPref.isFreqBands = 0;
3406 defPref.isFreqLog = 0;
3407 defPref.TimeBands = {};
3408 defPref.Freqs = [];
3409 defPref.FreqsLog = [];
3410 defPref.FreqBands = bst_get('DefaultFreqBands');
3411 defPref.Measure = 'power';
3412 defPref.Output = 'all';
3413 defPref.ClusterFuncTime = 'after';
3414 defPref.StftWinLen = 1;
3415 defPref.StftWinOvr = 0;
3416 defPref.StftFrqMax = 0;
3417 argout1 = FillMissingFields(contextName, defPref);
3418 if isempty(argout1.Freqs)
3419 argout1.Freqs = defPref.Freqs;
3420 end
3421 if ~isempty(argout1.FreqBands) && ~ischar(argout1.FreqBands{1,2})
3422 argout1.FreqBands = defPref.FreqBands;
3423 end
3424
3425 case 'ExportBidsOptions'
3426 defPref.ProjName = [];
3427 defPref.ProjID = [];
3428 defPref.ProjDesc = [];
3429 defPref.Categories = [];
3430 defPref.JsonDataset = ['{' 10 ' "License": "PD"' 10 '}'];
3431 defPref.JsonMeg = ['{' 10 ' "TaskDescription": "My task"' 10 '}'];
3432 argout1 = FillMissingFields(contextName, defPref);
3433
3434 case 'OpenMEEGOptions'
3435 defPref.BemFiles = {};
3436 defPref.BemNames = {'Scalp', 'Skull', 'Brain'};
3437 defPref.BemCond = [1, 0.0125, 1];
3438 defPref.BemSelect = [1 1 1];
3439 defPref.isAdjoint = 0;
3440 defPref.isAdaptative = 1;
3441 defPref.isSplit = 0;
3442 defPref.SplitLength = 4000;
3443 argout1 = FillMissingFields(contextName, defPref);
3444
3445 case 'DuneuroOptions'
3446 defPref = duneuro_defaults();
3447 argout1 = FillMissingFields(contextName, defPref);
3448
3449 case 'GridOptions_dipfit'
3450 defPref = struct(...
3451 'Method', 'isotropic', ...
3452 'nLayers', 17, ...
3453 'Reduction', 3, ...
3454 'nVerticesInit', 4000, ...
3455 'Resolution', 0.020, ...
3456 'FileName', '');
3457 argout1 = FillMissingFields(contextName, defPref);
3458
3459 case 'GridOptions_headmodel'
3460 defPref = struct(...
3461 'Method', 'isotropic', ...
3462 'nLayers', 17, ...
3463 'Reduction', 3, ...
3464 'nVerticesInit', 4000, ...
3465 'Resolution', 0.005, ...
3466 'FileName', '');
3467 argout1 = FillMissingFields(contextName, defPref);
3468
3469 case 'MriOptions'
3470 defPref = struct(...
3471 'isRadioOrient', 0, ...
3472 'isMipAnatomy', 0, ...
3473 'isMipFunctional', 0, ...
3474 'OverlaySmooth', 0, ...
3475 'InterpDownsample', 3, ...
3476 'DistanceThresh', 6, ...
3477 'UpsampleImage', 0, ...
3478 'DefaultAtlas', []);
3479 argout1 = FillMissingFields(contextName, defPref);
3480
3481 case 'DigitizeOptions'
3482 defPref = struct(...
3483 'PatientId', 'S001', ...
3484 'ComPort', 'COM1', ...
3485 'ComRate', 9600, ...
3486 'ComByteCount', 94, ... % 47 bytes * 2 receivers
3487 'UnitType', 'fastrak', ...
3488 'ConfigCommands', [], ... % setup-specific device configuration commands, e.g. hemisphere of operation
3489 'nFidSets', 2, ...
3490 'Fids', {{'NAS', 'LPA', 'RPA'}}, ... % 3 anat points (required) and any other, e.g. MEG coils, in desired digitization order
3491 'DistThresh', 0.005, ... % 5 mm distance threshold between repeated measures of fid positions
3492 'isBeep', 1, ...
3493 'isMEG', 1, ...
3494 'isSimulate', 0, ...
3495 'Montages', [...
3496 struct('Name', 'No EEG', ...
3497 'Labels', [], ...
3498 'ChannelFile', []), ...
3499 struct('Name', 'Default', ...
3500 'Labels', [], ...
3501 'ChannelFile', [])], ...
3502 'iMontage', 1, ...
3503 'Version', '2024'); % Version of the Digitize panel: 'legacy' or '2024'
3504 argout1 = FillMissingFields(contextName, defPref);
3505
3506 case 'PcaOptions'
3507 defPref.Method = 'pca'; % deprecated legacy per-file with sign inconsistencies, but kept as default for reproducibility
3508 defPref.Baseline = [-.1, 0]; % not used for 'pca': full window instead
3509 defPref.DataTimeWindow = [0, 1]; % not used for 'pca': full window instead
3510 defPref.RemoveDcOffset = 'file';
3511 argout1 = FillMissingFields(contextName, defPref);
3512
3513 case 'ConnectGraphOptions'
3514 % Get interface scaling factor
3515 InterfaceScaling = bst_get('InterfaceScaling');
3516 % Get default values
3517 defPref = struct(...
3518 'LobeFullLabel', 1, ...
3519 'TextDisplayMode', [1 2], ...
3520 'LabelSize', 7 .* InterfaceScaling ./ 100, ...
3521 'NodeSize', 5 .* InterfaceScaling ./ 100, ...
3522 'LinkSize', 1.5 .* InterfaceScaling ./ 100, ...
3523 'BgColor', [0 0 0], ...
3524 'HierarchyNodeIsVisible', 1);
3525 % If we have an additional argument, get the default values
3526 if nargin > 1
3527 argout1 = defPref;
3528 % Otherwise, get the saved values
3529 else
3530 savedValues = FillMissingFields(contextName, defPref);
3531
3532 % if any of the fields are [], replace by default value
3533 % do it here to avoid touching the common FillMissingFields
3534 % function, as other tools may actually want to set [] as desired property
3535 fields = fieldnames(savedValues);
3536 for i=1:numel(fields)
3537 if(isempty(savedValues.(fields{i})))
3538 savedValues.(fields{i}) = defPref.(fields{i});
3539 end
3540 end
3541 argout1 = savedValues;
3542 end
3543
3544 case 'NodelistOptions'
3545 defPref = struct(...
3546 'String', '', ... % What to search for
3547 'Target', 'Comment', ... % What field to search for: {'FileName', 'Comment'}
3548 'Action', 'Select'); % What to do with the filtered files: {'Select', 'Exclude'}
3549 argout1 = FillMissingFields(contextName, defPref);
3550
3551 case 'ReadOnly'
3552 if isfield(GlobalData.DataBase, 'isReadOnly')
3553 argout1 = GlobalData.DataBase.isReadOnly;
3554 else
3555 argout1 = 0;
3556 end
3557
3558 case 'LastPsdDisplayFunction'
3559 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'LastPsdDisplayFunction')
3560 argout1 = GlobalData.Preferences.LastPsdDisplayFunction;
3561 else
3562 argout1 = [];
3563 end
3564
3565 case 'LastTfDisplayFunction'
3566 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'LastTfDisplayFunction')
3567 argout1 = GlobalData.Preferences.LastTfDisplayFunction;
3568 else
3569 argout1 = [];
3570 end
3571
3572 case 'PlotlyCredentials'
3573 % Get saved username and API key
3574 try
3575 creds = loadplotlycredentials();
3576 argout1 = creds.username;
3577 argout2 = creds.api_key;
3578 catch
3579 argout1 = '';
3580 argout2 = '';
3581 end
3582 % Get saved domain
3583 try
3584 config = loadplotlyconfig();
3585 argout3 = config.plotly_domain;
3586 catch
3587 argout3 = '';
3588 end
3589
3590 case 'KlustersExecutable'
3591 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, 'KlustersExecutable')
3592 argout1 = GlobalData.Preferences.KlustersExecutable;
3593 else
3594 argout1 = [];
3595 end
3596
3597
3598 %% ===== FILE FILTERS =====
3599 case 'FileFilters'
3600 switch lower(varargin{2})
3601 case 'mri'
3602 argout1 = {...
3603 {'.img'}, 'MRI: Analyze (*.img/*.hdr)', 'Analyze'; ...
3604 {'.vmr'}, 'MRI: BESA (*.vmr)', 'BESA-VMR'; ...
3605 {'.ima'}, 'MRI: BrainVISA GIS (*.ima/*.dim)', 'GIS'; ...
3606 {'.mri'}, 'MRI: CTF (*.mri)', 'CTF'; ...
3607 {'.mat'}, 'MRI: FieldTrip (*.mat)', 'FT-MRI'; ...
3608 {'.mgh','.mgz'}, 'MRI: MGH (*.mgh,*.mgz)', 'MGH'; ...
3609 {'.mnc', '.mni'}, 'MRI: MNI (*.mnc,*.mni)', 'MINC'; ...
3610 {'.nii','.gz'}, 'MRI: NIfTI-1 (*.nii;*.nii.gz)', 'Nifti1'; ...
3611 {'.jnii','.bnii'}, 'MRI: JNIfTI (*.jnii;*.bnii)', 'JNIfTI'; ...
3612 {'_subjectimage'}, 'MRI: Brainstorm (*subjectimage*.mat)', 'BST'; ...
3613 {'*'}, 'MRI: DICOM (SPM converter)', 'DICOM-SPM'; ...
3614 {'.mri', '.fif', '.img', '.ima', '.nii', '.jnii', '.bnii', '.mgh', '.mgz', '.mnc', '.mni', '.gz', '_subjectimage'}, 'All MRI files (subject space)', 'ALL'; ...
3615 {'.mri', '.fif', '.img', '.ima', '.nii', '.jnii', '.bnii', '.mgh', '.mgz', '.mnc', '.mni', '.gz', '_subjectimage'}, 'All MRI files (MNI space)', 'ALL-MNI'; ...
3616 {'.mri', '.fif', '.img', '.ima', '.nii', '.jnii', '.bnii', '.mgh', '.mgz', '.mnc', '.mni', '.gz', '_subjectimage'}, 'Volume atlas (subject space)', 'ALL-ATLAS'; ...
3617 {'.mri', '.fif', '.img', '.ima', '.nii', '.jnii', '.bnii', '.mgh', '.mgz', '.mnc', '.mni', '.gz', '_subjectimage'}, 'Volume atlas (MNI space)', 'ALL-MNI-ATLAS'; ...
3618 };
3619 case 'mriout'
3620 argout1 = {...
3621 {'.img'}, 'MRI: Analyze (*.img/*.hdr)', 'Analyze'; ...
3622 {'.ima'}, 'MRI: BrainVISA GIS (*.ima/*.dim)', 'GIS'; ...
3623 {'.mri'}, 'MRI: CTF (*.mri)', 'CTF'; ...
3624 {'.mat'}, 'MRI: FieldTrip (*.mat)', 'FT-MRI'; ...
3625 {'.nii'}, 'MRI: NIfTI-1 (*.nii)', 'Nifti1';...
3626 {'.jnii','.bnii'}, 'MRI: JNIfTI (*.jnii;*.bnii)', 'JNIfTI';...
3627 };
3628 case 'anatin'
3629 argout1 = {...
3630 {'.folder'}, 'FreeSurfer', 'FreeSurfer-fast'; ...
3631 {'.folder'}, 'FreeSurfer + Volume atlases', 'FreeSurfer'; ...
3632 {'.folder'}, 'FreeSurfer + Volume atlases + Thickness', 'FreeSurfer+Thick'; ...
3633 {'.folder'}, 'BrainSuite', 'BrainSuite-fast'; ...
3634 {'.folder'}, 'BrainSuite + Volume atlases', 'BrainSuite'; ...
3635 {'.folder'}, 'BrainVISA', 'BrainVISA'; ...
3636 {'.folder'}, 'CAT12', 'CAT12'; ...
3637 {'.folder'}, 'CAT12 + Thickness', 'CAT12+Thick'; ...
3638 {'.folder'}, 'CIVET', 'CIVET'; ...
3639 {'.folder'}, 'CIVET + Thickness', 'CIVET+Thick'; ...
3640 {'.folder'}, 'HCP MEG/anatomy (pipeline v3)', 'HCPv3'; ...
3641 {'.folder'}, 'SimNIBS', 'SimNIBS'; ...
3642 {'.folder'}, 'BESA MRI', 'BESA-MRI'; ...
3643 };
3644 case 'source4d'
3645 argout1 = {...
3646 {'.folder'}, 'NIfTI-1 (*.nii)', 'Nifti1';...
3647 {'.folder'}, 'Analyze (*.img/*.hdr)', 'Analyze'; ...
3648 {'.folder'}, 'Matlab 4D matrix (*voltime*.mat)', 'BST'; ...
3649 };
3650
3651 case 'surface'
3652 argout1 = {...
3653 {'.mesh'}, 'BrainVISA (*.mesh)', 'MESH'; ...
3654 {'_tess', '_head', '_scalp', '_brain', '_cortex', '_innerskull', '_outerskull'}, 'Brainstorm (*.mat)', 'BST'; ...
3655 {'.dfs'}, 'BrainSuite (*.dfs)', 'DFS'; ...
3656 {'.dsgl'}, 'BrainSuite old (*.dsgl)', 'DSGL'; ...
3657 {'.bd0','.bd1','.bd2','.bd3','.bd4','.bd5','.bd6','.bd7','.bd8','.bd9', ...
3658 '.s00','.s01','.s02','.s03','.s04','.s05','.s06','.s07','.s08','.s09'}, ...
3659 'Curry BEM (*.db*;*.s0*)', 'CURRY-BEM';
3660 {'.vtk'}, 'FSL: VTK (*.vtk)', 'VTK'; ...
3661 {'*'}, 'FreeSurfer (*.*)', 'FS';
3662 {'.off'}, 'Geomview OFF (*.off)', 'OFF'; ...
3663 {'.gii'}, 'GIfTI / MRI coordinates (*.gii)', 'GII'; ...
3664 {'.gii'}, 'GIfTI / MNI coordinates (*.gii)', 'GII-MNI'; ...
3665 {'.gii'}, 'GIfTI / World coordinates (*.gii)', 'GII-WORLD'; ...
3666 {'.fif'}, 'MNE (*.fif)', 'FIF'; ...
3667 {'.obj'}, 'MNI OBJ (*.obj)', 'MNIOBJ'; ...
3668 {'.obj'}, 'Wavefront OBJ (*.obj)', 'WFTOBJ'; ...
3669 {'.srf'}, 'BESA (*.srf)', 'BESA-SRF'; ...
3670 {'.msh'}, 'SimNIBS3/headreco Gmsh4 (*.msh)', 'SIMNIBS3'; ...
3671 {'.msh'}, 'SimNIBS4/charm Gmsh4 (*.msh)', 'SIMNIBS4'; ...
3672 {'.tri'}, 'TRI (*.tri)', 'TRI'; ...
3673 {'.mri', '.fif', '.img', '.ima', '.nii', '.mgh', '.mgz', '.mnc', '.mni', '.gz', '_subjectimage'}, 'Volume mask or atlas (subject space)', 'MRI-MASK'; ...
3674 {'.mri', '.fif', '.img', '.ima', '.nii', '.mgh', '.mgz', '.mnc', '.mni', '.gz'}, 'Volume mask or atlas (MNI space)', 'MRI-MASK-MNI'; ...
3675 {'.nwbaux'}, 'Neurodata Without Borders (*.nwbaux)', 'NWB'; ...
3676 {'*'}, 'All surface files (*.*)', 'ALL'; ...
3677 };
3678
3679 case 'surfaceout'
3680 argout1 = {...
3681 {'.mesh'}, 'BrainVISA (*.mesh)', 'MESH'; ...
3682 {'.dfs'}, 'BrainSuite (*.dfs)', 'DFS'; ...
3683 {'.fs'}, 'FreeSurfer (*.fs)', 'FS'; ...
3684 {'.obj'}, 'Wavefront OBJ (*.obj)', 'OBJ'; ...
3685 {'.off'}, 'Geomview OFF (*.off)', 'OFF'; ...
3686 {'.gii'}, 'GIfTI (*.gii)', 'GII'; ...
3687 {'.tri'}, 'TRI (*.tri)', 'TRI'; ...
3688 {'.stl'}, 'STL (*.stl)', 'STL'; ...
3689 };
3690
3691 case 'data'
3692 argout1 = {...
3693 {'.*'}, 'MEG/EEG: 4D-Neuroimaging/BTi (*.*)', '4D'; ...
3694 {'.meg4','.res4'}, 'MEG/EEG: CTF (*.ds;*.meg4;*.res4)', 'CTF'; ...
3695 {'.fif'}, 'MEG/EEG: Elekta-Neuromag (*.fif)', 'FIF'; ...
3696 {'.mat'}, 'MEG/EEG: FieldTrip (*.mat)', 'FT-TIMELOCK'; ...
3697 {'.raw'}, 'MEG/EEG: ITAB (*.raw)', 'ITAB'; ...
3698 {'.kdf'}, 'MEG/EEG: KRISS MEG (*.kdf)', 'KDF'; ...
3699 {'.mrk','.sqd','.con','.raw','.ave'}, 'MEG/EEG: Ricoh (*.sqd;*.con;*.raw;*.ave;*.mrk)', 'RICOH'; ...
3700 {'.mat'}, 'MEG/EEG: SPM (*.mat/.dat)', 'SPM-DAT'; ...
3701 {'.mrk','.sqd','.con','.raw','.ave'}, 'MEG/EEG: Yokogawa/KIT (*.sqd;*.con;*.raw;*.ave;*.mrk)', 'KIT'; ...
3702 {'.meghdf5'}, 'MEG/EEG: York Instruments MEGSCAN (.meghdf5)', 'MEGSCAN-HDF5'; ...
3703 {'.bst'}, 'MEG/EEG: Brainstorm binary (*.bst)', 'BST-BIN'; ...
3704 {'.adicht'}, 'EEG: ADInstruments LabChart (*.adicht)', 'EEG-ADICHT'; ...
3705 {'.msr'}, 'EEG: ANT ASA (*.msr)', 'EEG-ANT-MSR'; ...
3706 {'.cnt','.avr'}, 'EEG: ANT EEProbe (*.cnt;*.avr)', 'EEG-ANT-CNT'; ...
3707 {'*'}, 'EEG: ASCII text (*.*)', 'EEG-ASCII'; ...
3708 {'.raw'}, 'EEG: Axion AxIS (*.raw)', 'EEG-AXION'; ...
3709 {'.dat'}, 'EEG: BCI2000 (*.dat)', 'EEG-BCI2000'; ...
3710 {'.bdf'}, 'EEG: BDF (*.bdf)', 'EEG-BDF'; ...
3711 {'.avr','.mux','.mul'}, 'EEG: BESA exports (*.avr;*.mul;*.mux)', 'EEG-BESA'; ...
3712 {'.acq'}, 'EEG: BIOPAC AcqKnowledge (*.acq)', 'EEG-BIOPAC'; ...
3713 {'.ns1','.ns2','.ns3','.ns4','.ns5','.ns6'}, 'EEG: Blackrock NeuroPort (*.nsX/*.nev)', 'EEG-BLACKROCK';
3714 {'.eeg','.dat'}, 'EEG: BrainVision BrainAmp (*.eeg;*.dat)', 'EEG-BRAINAMP'; ...
3715 {'.txt'}, 'EEG: BrainVision Analyzer (*.txt)', 'EEG-BRAINVISION'; ...
3716 {'.sef','.ep','.eph'}, 'EEG: Cartool (*.sef;*.ep;*.eph)', 'EEG-CARTOOL'; ...
3717 {'.dat','.cdt'}, 'EEG: Curry (*.dat;*.cdt)', 'EEG-CURRY'; ...
3718 {'.smr','.son'}, 'EEG: CED Spike2 old 32bit (*.smr;*.son)', 'EEG-SMR'; ...
3719 {'.smr','.smrx'}, 'EEG: CED Spike2 new 64bit (*.smr;*.smrx)', 'EEG-SMRX'; ...
3720 {'.rda'}, 'EEG: Compumedics ProFusion Sleep (*.rda)', 'EEG-COMPUMEDICS-PFS'; ...
3721 {'.bin'}, 'EEG: Deltamed Coherence-Neurofile (*.bin)', 'EEG-DELTAMED'; ...
3722 {'.edf','.rec'}, 'EEG: EDF / EDF+ (*.rec;*.edf)', 'EEG-EDF'; ...
3723 {'.edf','.rec'}, 'EEG EDF / EDF+ FieldTrip reader (*.rec;*.edf)', 'EEG-EDF-FT'; ...
3724 {'.set'}, 'EEG: EEGLAB (*.set)', 'EEG-EEGLAB'; ...
3725 {'.raw'}, 'EEG: EGI Netstation RAW (*.raw)', 'EEG-EGI-RAW'; ...
3726 {'.mff','.bin'}, 'EEG: EGI-Philips (*.mff)', 'EEG-EGI-MFF'; ...
3727 {'.edf'}, 'EEG: EmotivPRO (*.edf)', 'EEG-EMOTIV'; ...
3728 {'.erp','.hdr'}, 'EEG: ERPCenter (*.hdr;*.erp)', 'EEG-ERPCENTER'; ...
3729 {'.erp'}, 'EEG: ERPLab (*.erp)', 'EEG-ERPLAB'; ...
3730 {'.mat','.hdf5'}, 'EEG: g.tec Matlab (*.mat,*.hdf5)', 'EEG-GTEC'; ...
3731 {'.rhd','.rhs'}, 'EEG: Intan (*.rhd,*.rhs)', 'EEG-INTAN'; ...
3732 {'.mb2'}, 'EEG: MANSCAN (*.mb2)', 'EEG-MANSCAN'; ...
3733 {'.trc'}, 'EEG: Micromed (*.trc)', 'EEG-MICROMED'; ...
3734 {'.mat'}, 'EEG: Matlab matrix (*.mat)', 'EEG-MAT'; ...
3735 {'.csv'}, 'EEG: Muse (*.csv)', 'EEG-MUSE-CSV'; ...
3736 {'.ncs'}, 'EEG: Neuralynx (*.ncs)', 'EEG-NEURALYNX'; ...
3737 {'.nwb'}, 'EEG: Neurodata Without Borders (*.nwb)','NWB'; ...
3738 {'.nedf','.easy'}, 'EEG: Neuroelectrics (*.nedf;*.easy)', 'EEG-NEUROELECTRICS'; ...
3739 {'.bin'}, 'EEG: NeurOne session folder', 'EEG-NEURONE'; ...
3740 {'.cnt','.avg','.eeg','.dat'}, 'EEG: Neuroscan (*.cnt;*.eeg;*.avg;*.dat)', 'EEG-NEUROSCAN'; ...
3741 {'.eeg','.dat'}, 'EEG: NeuroScope (*.eeg;*.dat)', 'EEG-NEUROSCOPE'; ...
3742 {'.e'}, 'EEG: Nicolet (*.e)', 'EEG-NICOLET'; ...
3743 {'.eeg'}, 'EEG: Nihon Kohden (*.eeg)', 'EEG-NK'; ...
3744 {'.dat'}, 'EEG: Open Ephys flat binary (*.dat)', 'EEG-OEBIN'; ...
3745 {'.plx','.pl2'}, 'EEG: Plexon (*.plx;*.pl2)', 'EEG-PLEXON'; ...
3746 {'.ns1','.ns2','.ns3','.ns4','.ns5','.ns6'}, 'EEG: Ripple Trellis (*.nsX/*.nev)', 'EEG-RIPPLE'; ...
3747 {'.h5'}, 'EEG: The Virtual Brain (*_TimeSeriesEEG.h5)', 'EEG-TVB'; ...
3748 {'.csv'}, 'EEG: Wearable Sensing (*.csv)', 'EEG-WS-CSV'; ...
3749 {'.xdf'}, 'EEG: XDF (*.xdf)', 'EEG-XDF'; ...
3750 {'.nirs'}, 'NIRS: Brainsight (*.nirs)', 'NIRS-BRS'; ...
3751 {'.bnirs','.jnirs','.snirf'}, 'NIRS: SNIRF (*.snirf)', 'NIRS-SNIRF'; ...
3752 {'.edf'}, 'Eye tracker: EyeLink (*.edf)', 'EYELINK'; ...
3753 {'.tsv'}, 'Eye tracker: Tobii glasses (*.tsv)', 'EYE-TOBII-TSV'; ...
3754 };
3755 case 'raw'
3756 argout1 = {...
3757 {'.*'}, 'MEG/EEG: 4D-Neuroimaging/BTi (*.*)', '4D'; ...
3758 {'.meg4','.res4'}, 'MEG/EEG: CTF (*.ds;*.meg4;*.res4)', 'CTF'; ...
3759 {'.fif'}, 'MEG/EEG: Elekta-Neuromag (*.fif)', 'FIF'; ...
3760 {'.mat'}, 'MEG/EEG: FieldTrip (*.mat)', 'FT-TIMELOCK'; ...
3761 {'.raw'}, 'MEG/EEG: ITAB (*.raw)', 'ITAB'; ...
3762 {'.kdf'}, 'MEG/EEG: KRISS MEG (*.kdf)', 'KDF'; ...
3763 {'.mrk','.sqd','.con','.raw','.ave'}, 'MEG/EEG: Ricoh (*.sqd;*.con;*.raw;*.ave;*.mrk)', 'RICOH'; ...
3764 {'.mat'}, 'MEG/EEG: SPM (*.mat/.dat)', 'SPM-DAT'; ...
3765 {'.mrk','.sqd','.con','.raw','.ave'}, 'MEG/EEG: Yokogawa/KIT (*.sqd;*.con;*.raw;*.ave;*.mrk)', 'KIT'; ...
3766 {'.meghdf5'}, 'MEG/EEG: York Instruments MEGSCAN (.meghdf5)', 'MEGSCAN-HDF5'; ...
3767 {'.bst'}, 'MEG/EEG: Brainstorm binary (*.bst)', 'BST-BIN'; ...
3768 {'.adicht'}, 'EEG: ADInstruments LabChart (*.adicht)', 'EEG-ADICHT'; ...
3769 {'.msr'}, 'EEG: ANT ASA (*.msr)', 'EEG-ANT-MSR'; ...
3770 {'.cnt','.avr'}, 'EEG: ANT EEProbe (*.cnt;*.avr)', 'EEG-ANT-CNT'; ...
3771 {'*'}, 'EEG: ASCII text (*.*)', 'EEG-ASCII'; ...
3772 {'.raw'}, 'EEG: Axion AxIS (*.raw)', 'EEG-AXION'; ...
3773 {'.dat'}, 'EEG: BCI2000 (*.dat)', 'EEG-BCI2000'; ...
3774 {'.bdf'}, 'EEG: BDF (*.bdf)', 'EEG-BDF'; ...
3775 {'.avr','.mux','.mul'}, 'EEG: BESA exports (*.avr;*.mul;*.mux)', 'EEG-BESA'; ...
3776 {'.acq'}, 'EEG: BIOPAC AcqKnowledge (*.acq)', 'EEG-BIOPAC'; ...
3777 {'.ns1','.ns2','.ns3','.ns4','.ns5','.ns6'}, 'EEG: Blackrock NeuroPort (*.nsX/*.nev)', 'EEG-BLACKROCK';
3778 {'.eeg','.dat'}, 'EEG: BrainVision BrainAmp (*.eeg;*.dat)', 'EEG-BRAINAMP'; ...
3779 {'.txt'}, 'EEG: BrainVision Analyzer (*.txt)', 'EEG-BRAINVISION'; ...
3780 {'.sef','.ep','.eph'}, 'EEG: Cartool (*.sef;*.ep;*.eph)', 'EEG-CARTOOL'; ...
3781 {'.smr','.son'}, 'EEG: CED Spike2 old 32bit (*.smr;*.son)', 'EEG-SMR'; ...
3782 {'.smr','.smrx'}, 'EEG: CED Spike2 new 64bit (*.smr;*.smrx)', 'EEG-SMRX'; ...
3783 {'.rda'}, 'EEG: Compumedics ProFusion Sleep (*.rda)', 'EEG-COMPUMEDICS-PFS'; ...
3784 {'.dat','.cdt'}, 'EEG: Curry (*.dat;*.cdt)', 'EEG-CURRY'; ...
3785 {'.bin'}, 'EEG: Deltamed Coherence-Neurofile (*.bin)', 'EEG-DELTAMED'; ...
3786 {'.edf','.rec'}, 'EEG: EDF / EDF+ (*.rec;*.edf)', 'EEG-EDF'; ...
3787 {'.edf','.rec'}, 'EEG EDF / EDF+ FieldTrip reader (*.rec;*.edf)', 'EEG-EDF-FT'; ...
3788 {'.set'}, 'EEG: EEGLAB (*.set)', 'EEG-EEGLAB'; ...
3789 {'.raw'}, 'EEG: EGI Netstation RAW (*.raw)', 'EEG-EGI-RAW'; ...
3790 {'.mff','.bin'}, 'EEG: EGI-Philips (*.mff)', 'EEG-EGI-MFF'; ...
3791 {'.edf'}, 'EEG: EmotivPRO (*.edf)', 'EEG-EMOTIV'; ...
3792 {'.mat','.hdf5'}, 'EEG: g.tec Matlab (*.mat,*.hdf5)', 'EEG-GTEC'; ...
3793 {'.rhd','.rhs'}, 'EEG: Intan (*.rhd,*.rhs)', 'EEG-INTAN'; ...
3794 {'.mb2'}, 'EEG: MANSCAN (*.mb2)', 'EEG-MANSCAN'; ...
3795 {'.mat'}, 'EEG: Matlab matrix (*.mat)', 'EEG-MAT'; ...
3796 {'.csv'}, 'EEG: Muse (*.csv)', 'EEG-MUSE-CSV'; ...
3797 {'.trc'}, 'EEG: Micromed (*.trc)', 'EEG-MICROMED'; ...
3798 {'.ncs'}, 'EEG: Neuralynx (*.ncs)', 'EEG-NEURALYNX'; ...
3799 {'.nwb'}, 'EEG: Neurodata Without Borders (*.nwb)','NWB'; ...
3800 {'.nedf','.easy'}, 'EEG: Neuroelectrics (*.nedf;*.easy)', 'EEG-NEUROELECTRICS'; ...
3801 {'.bin'}, 'EEG: NeurOne session folder', 'EEG-NEURONE'; ...
3802 {'.cnt','.avg','.eeg','.dat'}, 'EEG: Neuroscan (*.cnt;*.eeg;*.avg;*.dat)', 'EEG-NEUROSCAN'; ...
3803 {'.eeg','.dat'}, 'EEG: NeuroScope (*.eeg;*.dat)', 'EEG-NEUROSCOPE'; ...
3804 {'.e'}, 'EEG: Nicolet (*.e)', 'EEG-NICOLET'; ...
3805 {'.eeg'}, 'EEG: Nihon Kohden (*.eeg)', 'EEG-NK'; ...
3806 {'.dat'}, 'EEG: Open Ephys flat binary (*.dat)', 'EEG-OEBIN'; ...
3807 {'.plx','.pl2'}, 'EEG: Plexon (*.plx;.pl2)' 'EEG-PLEXON'; ...
3808 {'.ns1','.ns2','.ns3','.ns4','.ns5','.ns6'}, 'EEG: Ripple Trellis (*.nsX/*.nev)', 'EEG-RIPPLE'; ...
3809 {'.h5'}, 'EEG: The Virtual Brain (*_TimeSeriesEEG.h5)', 'EEG-TVB'; ...
3810 {'.tbk'}, 'EEG: Tucker Davis Technologies (*.tbk)', 'EEG-TDT'; ...
3811 {'.csv'}, 'EEG: Wearable Sensing (*.csv)', 'EEG-WS-CSV'; ...
3812 {'.xdf'}, 'EEG: XDF (*.xdf)', 'EEG-XDF'; ...
3813 {'.trc','.eeg','.e','.bin','.rda','.edf','.bdf'}, 'SEEG: Deltamed/Micromed/NK/Nicolet/BrainVision/EDF', 'SEEG-ALL'; ...
3814 {'.trc','.eeg','.e','.bin','.rda','.edf','.bdf'}, 'ECOG: Deltamed/Micromed/NK/Nicolet/BrainVision/EDF', 'ECOG-ALL'; ...
3815 {'.nirs'}, 'NIRS: Brainsight (*.nirs)', 'NIRS-BRS'; ...
3816 {'.bnirs','.jnirs','.snirf'}, 'NIRS: SNIRF (*.snirf)', 'NIRS-SNIRF'; ...
3817 {'.edf'}, 'Eye tracker: EyeLink (*.edf)', 'EYELINK'; ...
3818 {'.tsv'}, 'Eye tracker: Tobii glasses (*.tsv)', 'EYE-TOBII-TSV'; ...
3819 };
3820
3821 case 'dataout'
3822 argout1 = {...
3823 {'.bst'}, 'MEG/EEG: Brainstorm binary (*.bst)', 'BST-BIN'; ...
3824 {'.mat'}, 'MEG/EEG: FieldTrip timelock (*.mat)', 'FT-TIMELOCK'; ...
3825 {'.mat'}, 'MEG/EEG: SPM (*.mat/.dat)', 'SPM-DAT'; ...
3826 {'.eeg'}, 'EEG: BrainVision BrainAmp (*.eeg)', 'EEG-BRAINAMP'; ...
3827 {'.eph'}, 'EEG: Cartool EPH (*.eph)', 'EEG-CARTOOL-EPH'; ...
3828 {'.edf'}, 'EEG: EDF+ (*.edf)', 'EEG-EDF'; ...
3829 {'.raw'}, 'EEG: EGI NetStation RAW (*.raw)', 'EEG-EGI-RAW'; ...
3830 {'.snirf'}, 'NIRS: SNIRF (*.snirf)', 'NIRS-SNIRF'; ...
3831 {'.txt'}, 'ASCII: Space-separated, fixed column size (*.txt)', 'ASCII-SPC'; ...
3832 {'.txt'}, 'ASCII: Space-separated with header, fixed column size (*.txt)', 'ASCII-SPC-HDR'; ...
3833 {'.tsv'}, 'ASCII: Tab-separated (*.tsv)', 'ASCII-TSV'; ...
3834 {'.tsv'}, 'ASCII: Tab-separated with header (*.tsv)', 'ASCII-TSV-HDR'; ...
3835 {'.tsv'}, 'ASCII: Tab-separated with header transposed (*.tsv)', 'ASCII-TSV-HDR-TR'; ...
3836 {'.csv'}, 'ASCII: Comma-separated (*.csv)', 'ASCII-CSV'; ...
3837 {'.csv'}, 'ASCII: Comma-separated with header (*.csv)', 'ASCII-CSV-HDR'; ...
3838 {'.csv'}, 'ASCII: Comma-separated with header transposed (*.csv)', 'ASCII-CSV-HDR-TR'; ...
3839 {'.xlsx'}, 'Microsoft Excel (*.xlsx)', 'EXCEL'; ...
3840 {'.xlsx'}, 'Microsoft Excel transposed (*.xlsx)', 'EXCEL-TR'; ...
3841 {'_timeseries'}, 'Brainstorm matrix (*timeseries*.mat)', 'BST'; ...
3842 };
3843 case 'rawout'
3844 argout1 = {...
3845 {'.bst'}, 'MEG/EEG: Brainstorm binary (*.bst)', 'BST-BIN'; ...
3846 {'.mat'}, 'MEG/EEG: SPM (*.mat/.dat)', 'SPM-DAT'; ...
3847 {'.eeg'}, 'EEG: BrainVision BrainAmp (*.eeg)', 'EEG-BRAINAMP'; ...
3848 {'.edf'}, 'EEG: EDF+ (*.edf)', 'EEG-EDF'; ...
3849 {'.raw'}, 'EEG: EGI NetStation RAW (*.raw)', 'EEG-EGI-RAW'; ...
3850 {'.snirf'}, 'NIRS: SNIRF (*.snirf)', 'NIRS-SNIRF'; ...
3851 };
3852 case 'events'
3853 argout1 = {...
3854 {'.trg'}, 'ANT EEProbe (*.trg)', 'ANT'; ...
3855 {'.mrk'}, 'AnyWave (*.mrk)', 'ANYWAVE'; ...
3856 {'.evt'}, 'BESA (*.evt)', 'BESA'; ...
3857 {'_events.tsv'}, 'BIDS events: onset, duration, trial_type, channel (*_events.tsv)', 'BIDS'; ...
3858 {'.vmrk'}, 'BrainVision BrainAmp (*.vmrk)', 'BRAINAMP'; ...
3859 {'_events'}, 'Brainstorm (events*.mat)', 'BST'; ...
3860 {'.mrk'}, 'Cartool (*.mrk)', 'CARTOOL'; ...
3861 {'.mrk'}, 'CTF MarkerFile (*.mrk)', 'CTF'; ...
3862 {'.cef'}, 'Curry (*.cef)', 'CURRY'; ...
3863 {'.eve','.fif'}, 'Elekta-Neuromag MNE (*.eve;*.fif)', 'FIF'; ...
3864 {'.evl','.txt'}, 'Elekta-Neuromag Graph (*.evl;*.txt)', 'GRAPH'; ...
3865 {'.txt','.mat'}, 'FieldTrip trial definition (*.txt;*.mat)', 'TRL'; ...
3866 {'.trg'}, 'KRISS MEG (*.trg)', 'KDF'; ...
3867 {'.evt'}, 'Micromed (*.evt)', 'MICROMED'; ...
3868 {'.ev2'}, 'Neuroscan (*.ev2)', 'NEUROSCAN'; ...
3869 {'.txt'}, 'Nicolet export (*.txt)', 'NICOLET'; ...
3870 {'timestamps.npy'},'Open Ephys (timestamps.npy)', 'OEBIN'; ...
3871 {'.log'}, 'Presentation (*.log)', 'PRESENTATION'; ...
3872 {'.mrk','.sqd','.con','.raw','.ave'}, 'Ricoh (*.mrk;*.sqd;*.con;*.raw;*.ave)', 'RICOH'; ...
3873 {'.txt'}, 'XLTEK export (*.txt)', 'XLTEK'; ...
3874 {'.mrk','.sqd','.con','.raw','.ave'}, 'Yokogawa/KIT (*.mrk;*.sqd;*.con;*.raw;*.ave)', 'KIT'; ...
3875 {'.*'}, 'Array of times (*.mat;*.*)', 'ARRAY-TIMES'; ...
3876 {'.*'}, 'Array of samples (*.mat;*.*)', 'ARRAY-SAMPLES'; ...
3877 {'.txt','.csv'}, 'CSV text file: label, time, duration (*.txt;*.csv)', 'CSV-TIME'; ...
3878 {'.*'}, 'CTF Video Times (.txt)', 'CTFVIDEO'; ...
3879 };
3880 case 'eventsout'
3881 argout1 = {...
3882 {'.mrk'}, 'AnyWave (*.mrk)', 'ANYWAVE'; ...
3883 {'_events'}, 'Brainstorm (events*.mat)', 'BST'; ...
3884 {'.vmrk'}, 'BrainVision BrainAmp (*.vmrk)', 'BRAINAMP'; ...
3885 {'.mrk'}, 'CTF MarkerFile (*.mrk)', 'CTF'; ...
3886 {'.eve','.fif'}, 'Elekta-Neuromag/MNE (*.eve)', 'FIF'; ...
3887 {'.evl'}, 'Elekta-Neuromag Graph (Alternative Style) (*.evl)', 'GRAPH_ALT'; ...
3888 {'.txt'}, 'Array of times (*.txt)', 'ARRAY-TIMES'; ...
3889 {'.txt'}, 'Array of samples (*.txt)', 'ARRAY-SAMPLES'; ...
3890 {'.txt','.csv'}, 'CSV text file: label, time, duration (*.txt;*.csv)', 'CSV-TIME'; ...
3891 {'.txt'}, 'CTF Video Times (*.txt)', 'CTFVIDEO'; ...
3892 {'_events.tsv'}, 'BIDS events: onset, duration, trial_type, channel (*_events.tsv)', 'BIDS'; ...
3893 };
3894 case 'channel'
3895 argout1 = {...
3896 {'.*'}, 'MEG/EEG: 4D-Neuroimaging/BTi (*.*)', '4D'; ...
3897 {'.meg4','.res4'}, 'MEG/EEG: CTF (*.ds;*.meg4;*.res4)', 'CTF' ; ...
3898 {'.fif'}, 'MEG/EEG: Elekta-Neuromag (*.fif)', 'FIF'; ...
3899 {'.kdf'}, 'MEG/EEG: KRISS MEG (*.kdf)', 'KDF'; ...
3900 {'.raw'}, 'MEG/EEG: ITAB (*.raw)', 'ITAB'; ...
3901 {'.mrk','.sqd','.con','.raw','.ave'}, 'MEG/EEG: Ricoh (*.sqd;*.con;*.raw;*.ave;*.mrk)', 'RICOH'; ...
3902 {'.mrk','.sqd','.con','.raw','.ave'}, 'MEG/EEG: Yokogawa/KIT (*.sqd;*.con;*.raw;*.ave;*.mrk)', 'KIT'; ...
3903 {'.meghdf5'}, 'MEG/EEG: York Instruments MEGSCAN (.meghdf5)', 'MEGSCAN-HDF5'; ...
3904 {'.bst'}, 'MEG/EEG: Brainstorm binary (*.bst)', 'BST-BIN'; ...
3905 {'_channel'}, 'MEG/EEG: Brainstorm (channel*.mat)', 'BST'; ...
3906 {'.elc'}, 'EEG: ANT ASA/Xensor (*.elc)', 'XENSOR'; ...
3907 {'.sfp','.elp','.ela','.eps'}, 'EEG: BESA (*.sfp;*.elp;*.eps/*.ela)', 'BESA'; ...
3908 {'.bvef','.bvct','.txt'}, 'EEG: BrainVision electrode file (*.bvef,*.bvct,*.txt)', 'BRAINVISION'; ...
3909 {'.tsv'}, 'EEG: BIDS electrodes.tsv, subject space mm (*.tsv)', 'BIDS-SCANRAS-MM'; ...
3910 {'.tsv'}, 'EEG: BIDS electrodes.tsv, MNI space mm (*.tsv)', 'BIDS-MNI-MM'; ...
3911 {'.tsv'}, 'EEG: BIDS electrodes.tsv, ACPC space mm (*.tsv)', 'BIDS-ACPC-MM'; ...
3912 {'.tsv'}, 'EEG: BIDS electrodes.tsv, ALS/SCS/CTF space mm (*.tsv)','BIDS-ALS-MM'; ...
3913 {'.tsv'}, 'EEG: BIDS electrodes.tsv, CapTrak space mm (*.tsv)', 'BIDS-CAPTRAK-MM'; ...
3914 {'.els','.xyz'}, 'EEG: Cartool (*.els;*.xyz)', 'CARTOOL'; ...
3915 {'.eeg'}, 'EEG: MegDraw (*.eeg)', 'MEGDRAW'; ...
3916 {'.res','.rs3','.pom'}, 'EEG: Curry, LPS (*.res;*.rs3;*.pom)', 'CURRY'; ...
3917 {'.ced','.xyz','.set'}, 'EEG: EEGLAB (*.ced;*.xyz;*.set)', 'EEGLAB'; ...
3918 {'.elc'}, 'EEG: EETrak (*.elc)', 'EETRAK'; ...
3919 {'.sfp'}, 'EEG: EGI (*.sfp)', 'EGI'; ...
3920 {'coordinates.xml'}, 'EEG: EGI-Philips (coordinates.xml)', 'MFF'; ...
3921 {'.elp'}, 'EEG: EMSE (*.elp)', 'EMSE'; ...
3922 {'.pts','.csv'}, 'EEG: IntrAnat, subject space (*.pts;*.csv)', 'INTRANAT'; ...
3923 {'.pts','.csv'}, 'EEG: IntrAnat, MNI space (*.pts;*.csv)', 'INTRANAT_MNI'; ...
3924 {'.csv'}, 'EEG: Localite (*.csv)', 'LOCALITE'; ...
3925 {'.dat','.tri','.txt','.asc'}, 'EEG: Neuroscan (*.dat;*.tri;*.txt;*.asc)', 'NEUROSCAN'; ...
3926 {'.pos','.pol','.elp','.txt'}, 'EEG: Polhemus (*.pos;*.pol;*.elp;*.txt)', 'POLHEMUS'; ...
3927 {'.csv'}, 'EEG: SimNIBS (*.csv)', 'SIMNIBS'; ...
3928 {'.h5'}, 'EEG: The Virtual Brain (*_SensorsEEG.h5)', 'TVB'; ...
3929 {'*'}, 'EEG: ASCII: Name,XYZ (*.*)', 'ASCII_NXYZ'; ...
3930 {'*'}, 'EEG: ASCII: Name,XYZ_MNI (*.*)', 'ASCII_NXYZ_MNI'; ...
3931 {'*'}, 'EEG: ASCII: Name,XYZ_World (*.*)', 'ASCII_NXYZ_WORLD'; ...
3932 {'*'}, 'EEG: ASCII: Name,XY (*.*)', 'ASCII_NXY'; ...
3933 {'*'}, 'EEG: ASCII: XYZ (*.*)', 'ASCII_XYZ'; ...
3934 {'*'}, 'EEG: ASCII: XYZ_MNI (*.*)', 'ASCII_XYZ_MNI'; ...
3935 {'*'}, 'EEG: ASCII: XYZ_World (*.*)', 'ASCII_XYZ_WORLD'; ...
3936 {'*'}, 'EEG: ASCII: XY (*.*)', 'ASCII_XY'; ...
3937 {'*'}, 'EEG: ASCII: XYZ,Name (*.*)', 'ASCII_XYZN'; ...
3938 {'*'}, 'EEG: ASCII: XYZ_MNI,Name (*.*)', 'ASCII_XYZN_MNI'; ...
3939 {'*'}, 'EEG: ASCII: XYZ_World,Name (*.*)', 'ASCII_XYZN_WORLD'; ...
3940 {'*'}, 'EEG: ASCII: Name,Theta,Phi (*.*)', 'ASCII_NTP'; ...
3941 {'*'}, 'EEG: ASCII: Theta,Phi (*.*)', 'ASCII_TP'; ...
3942 };
3943 case 'channelout'
3944 argout1 = {...
3945 {'.pos'}, 'EEG+Headshape: Polhemus (*.pos)', 'POLHEMUS'; ...
3946 {'.eeg'}, 'Headshape: MegDraw (*.eeg)', 'MEGDRAW'; ...
3947 {'.pos'}, 'Headshape: Polhemus (*.pos)', 'POLHEMUS-HS'; ...
3948 {'.txt'}, 'Headshape: ASCII: XYZ (*.txt)', 'ASCII_XYZ-HS'; ...
3949 {'.txt'}, 'Headshape: ASCII: XYZ_World (*.txt)', 'ASCII_XYZ_WORLD-HS'; ...
3950 {'.txt'}, 'Headshape: ASCII: Name,XYZ (*.txt)', 'ASCII_NXYZ-HS'; ...
3951 {'.txt'}, 'Headshape: ASCII: Name,XYZ_World (*.txt)', 'ASCII_NXYZ_WORLD-HS'; ...
3952 {'.txt'}, 'Headshape: ASCII: XYZ,Name (*.txt)', 'ASCII_XYZN-HS'; ...
3953 {'.txt'}, 'Headshape: ASCII: XYZ_World,Name (*.txt)', 'ASCII_XYZN_WORLD-HS'; ...
3954 {'.sfp'}, 'EEG: BESA (*.sfp)', 'BESA-SFP'; ...
3955 {'.elp'}, 'EEG: BESA (*.elp)', 'BESA-ELP'; ...
3956 {'.tsv'}, 'EEG: BIDS electrodes.tsv, subject space mm (*.tsv)', 'BIDS-SCANRAS-MM'; ...
3957 {'.tsv'}, 'EEG: BIDS electrodes.tsv, MNI space mm (*.tsv)', 'BIDS-MNI-MM'; ...
3958 {'.tsv'}, 'EEG: BIDS electrodes.tsv, ALS/SCS/CTF space mm (*.tsv)', 'BIDS-ALS-MM'; ...
3959 {'.xyz'}, 'EEG: Cartool (*.xyz)', 'CARTOOL-XYZ'; ...
3960 {'.res'}, 'EEG: Curry (*.res)', 'CURRY-RES'; ...
3961 {'.xyz'}, 'EEG: EEGLAB (*.xyz)', 'EEGLAB-XYZ'; ...
3962 {'.sfp'}, 'EEG: EGI (*.sfp)', 'EGI'; ...
3963 {'.txt'}, 'EEG/NIRS: ASCII: XYZ (*.txt)', 'ASCII_XYZ-EEG'; ...
3964 {'.txt'}, 'EEG/NIRS: ASCII: XYZ_MNI (*.txt)', 'ASCII_XYZ_MNI-EEG'; ...
3965 {'.txt'}, 'EEG/NIRS: ASCII: XYZ_World (*.txt)', 'ASCII_XYZ_WORLD-EEG'; ...
3966 {'.txt'}, 'EEG/NIRS: ASCII: Name,XYZ (*.txt)', 'ASCII_NXYZ-EEG'; ...
3967 {'.txt'}, 'EEG/NIRS: ASCII: Name,XYZ_MNI (*.txt)', 'ASCII_NXYZ_MNI-EEG'; ...
3968 {'.txt'}, 'EEG/NIRS: ASCII: Name,XYZ_World (*.txt)', 'ASCII_NXYZ_WORLD-EEG'; ...
3969 {'.txt'}, 'EEG/NIRS: ASCII: XYZ,Name (*.txt)', 'ASCII_XYZN-EEG'; ...
3970 {'.txt'}, 'EEG/NIRS: ASCII: XYZ_MNI,Name (*.txt)', 'ASCII_XYZN_MNI-EEG'; ...
3971 {'.txt'}, 'EEG/NIRS: ASCII: XYZ_World,Name (*.txt)', 'ASCII_XYZN_WORLD-EEG'; ...
3972 {'.txt'}, 'EEG/NIRS: Brainsight (*.txt)', 'BRAINSIGHT-TXT'; ...
3973 {'.tsv'}, 'NIRS: BIDS optrodes.tsv, subject space mm (*.tsv)', 'BIDS-NIRS-SCANRAS-MM'; ...
3974 {'.tsv'}, 'NIRS: BIDS optrodes.tsv, MNI space mm (*.tsv)', 'BIDS-NIRS-MNI-MM'; ...
3975 {'.tsv'}, 'NIRS: BIDS optrodes.tsv, ALS/SCS/CTF space mm (*.tsv)', 'BIDS-NIRS-ALS-MM'; ...
3976 };
3977 case 'labelin'
3978 argout1 = {...
3979 {'.dfs'}, 'BrainSuite atlas (*.dfs)', 'DFS'; ...
3980 {'.annot'}, 'FreeSurfer atlas (*.annot)', 'FS-ANNOT'; ...
3981 {'.label'}, 'FreeSurfer ROI, single scout (*.label)', 'FS-LABEL-SINGLE'; ...
3982 {'.label'}, 'FreeSurfer ROI, probability map (*.label)', 'FS-LABEL'; ...
3983 {'.gii'}, 'GIfTI texture (*.gii)', 'GII-TEX'; ...
3984 {'.dset'}, 'SUMA atlas (*.dset)', 'DSET'; ...
3985 {'_scout'}, 'Brainstorm scouts (*scout*.mat)', 'BST'; ...
3986 {'.mri', '.fif', '.img', '.ima', '.nii', '.mgh', '.mgz', '.mnc', '.mni', '.gz', '_subjectimage'}, 'Volume mask or atlas (dilated, subject space)', 'MRI-MASK'; ...
3987 {'.mri', '.fif', '.img', '.ima', '.nii', '.mgh', '.mgz', '.mnc', '.mni', '.gz'}, 'Volume mask or atlas (dilated, MNI space)', 'MRI-MASK-MNI'; ...
3988 {'.mri', '.fif', '.img', '.ima', '.nii', '.mgh', '.mgz', '.mnc', '.mni', '.gz', '_subjectimage'}, 'Volume mask or atlas (no overlap, subject space)', 'MRI-MASK-NOOVERLAP'; ...
3989 {'.mri', '.fif', '.img', '.ima', '.nii', '.mgh', '.mgz', '.mnc', '.mni', '.gz'}, 'Volume mask or atlas (no overlap, MNI space)', 'MRI-MASK-NOOVERLAP-MNI'; ...
3990 };
3991 case 'resultsout'
3992 argout1 = {...
3993 {'_sources'}, 'Brainstorm sources (*sources*.mat)', 'BST'; ...
3994 {'.mat'}, 'FieldTrip sources (*.mat)', 'FT-SOURCES'; ...
3995 {'.txt'}, 'ASCII: Space-separated, fixed columns size (*.txt)', 'ASCII-SPC'; ...
3996 {'.txt'}, 'ASCII: Space-separated with header, fixed column size (*.txt)', 'ASCII-SPC-HDR'; ...
3997 {'.tsv'}, 'ASCII: Tab-separated (*.tsv)', 'ASCII-TSV'; ...
3998 {'.tsv'}, 'ASCII: Tab-separated with header (*.tsv)', 'ASCII-TSV-HDR'; ...
3999 {'.tsv'}, 'ASCII: Tab-separated with header transposed (*.tsv)', 'ASCII-TSV-HDR-TR'; ...
4000 {'.csv'}, 'ASCII: Comma-separated (*.csv)', 'ASCII-CSV'; ...
4001 {'.csv'}, 'ASCII: Comma-separated with header (*.csv)', 'ASCII-CSV-HDR'; ...
4002 {'.csv'}, 'ASCII: Comma-separated with header transposed (*.csv)', 'ASCII-CSV-HDR-TR'; ...
4003 {'.xlsx'}, 'Microsoft Excel (*.xlsx)', 'EXCEL'; ...
4004 {'.xlsx'}, 'Microsoft Excel transposed (*.xlsx)', 'EXCEL-TR'; ...
4005 };
4006 case 'timefreqout'
4007 argout1 = {...
4008 {'_timefreq'}, 'Brainstorm structure (*timefreq*.mat)', 'BST'; ...
4009 {'.mat'}, 'FieldTrip freq (*.mat)', 'FT-FREQ'; ...
4010 {'.txt'}, 'ASCII: Space-separated, fixed columns size (*.txt)', 'ASCII-SPC'; ...
4011 {'.txt'}, 'ASCII: Space-separated with header, fixed column size (*.txt)', 'ASCII-SPC-HDR'; ...
4012 {'.tsv'}, 'ASCII: Tab-separated (*.tsv)', 'ASCII-TSV'; ...
4013 {'.tsv'}, 'ASCII: Tab-separated with header (*.tsv)', 'ASCII-TSV-HDR'; ...
4014 {'.tsv'}, 'ASCII: Tab-separated with header transposed (*.tsv)', 'ASCII-TSV-HDR-TR'; ...
4015 {'.csv'}, 'ASCII: Comma-separated (*.csv)', 'ASCII-CSV'; ...
4016 {'.csv'}, 'ASCII: Comma-separated with header (*.csv)', 'ASCII-CSV-HDR'; ...
4017 {'.csv'}, 'ASCII: Comma-separated with header transposed (*.csv)', 'ASCII-CSV-HDR-TR'; ...
4018 {'.xlsx'}, 'Microsoft Excel (*.xlsx)', 'EXCEL'; ...
4019 {'.xlsx'}, 'Microsoft Excel transposed (*.xlsx)', 'EXCEL-TR'; ...
4020 };
4021 case 'matrixout'
4022 argout1 = {...
4023 {'_matrix'}, 'Brainstorm structure (*matrix*.mat)', 'BST'; ...
4024 {'.bst'}, 'MEG/EEG: Brainstorm binary (*.bst)', 'BST-BIN'; ...
4025 {'.mat'}, 'FieldTrip timelock (*.mat)', 'FT-TIMELOCK'; ...
4026 {'.edf'}, 'EEG: EDF+ (*.edf)', 'EEG-EDF'; ...
4027 {'.txt'}, 'ASCII: Space-separated, fixed columns size (*.txt)', 'ASCII-SPC'; ...
4028 {'.txt'}, 'ASCII: Space-separated with header, fixed column size (*.txt)', 'ASCII-SPC-HDR'; ...
4029 {'.tsv'}, 'ASCII: Tab-separated (*.tsv)', 'ASCII-TSV'; ...
4030 {'.tsv'}, 'ASCII: Tab-separated with header (*.tsv)', 'ASCII-TSV-HDR'; ...
4031 {'.tsv'}, 'ASCII: Tab-separated with header transposed (*.tsv)', 'ASCII-TSV-HDR-TR'; ...
4032 {'.csv'}, 'ASCII: Comma-separated (*.csv)', 'ASCII-CSV'; ...
4033 {'.csv'}, 'ASCII: Comma-separated with header (*.csv)', 'ASCII-CSV-HDR'; ...
4034 {'.csv'}, 'ASCII: Comma-separated with header transposed (*.csv)', 'ASCII-CSV-HDR-TR'; ...
4035 {'.xlsx'}, 'Microsoft Excel (*.xlsx)', 'EXCEL'; ...
4036 {'.xlsx'}, 'Microsoft Excel transposed (*.xlsx)', 'EXCEL-TR'; ...
4037 };
4038 case 'montagein'
4039 argout1 = {...
4040 {'.sel'}, 'MNE selection files (*.sel)', 'MNE'; ...
4041 {'.mon'}, 'Text montage files (*.mon)', 'MON'; ...
4042 {'_montage'}, 'Brainstorm montage files (montage_*.mat)', 'BST';
4043 {'.csv'}, 'Comma-separated montage files (*.csv)', 'CSV'};
4044 case 'montageout'
4045 argout1 = {...
4046 {'.sel'}, 'MNE selection files (*.sel)', 'MNE'; ...
4047 {'.mon'}, 'Text montage files (*.mon)', 'MON'; ...
4048 {'_montage'}, 'Brainstorm montage files (montage_*.mat)', 'BST'};
4049 case 'clusterin'
4050 argout1 = {...
4051 {'_cluster'}, 'Brainstorm clusters file (*cluster*.mat)', 'BST'; ...
4052 {'.sel'}, 'MNE selection files (*.sel)', 'MNE'};
4053 case 'fibers'
4054 argout1 = {...
4055 {'.trk'}, 'TrackVis (*.trk)', 'TRK'; ...
4056 {'_fibers'}, 'Brainstorm fibers files (fibers_*.mat)', 'BST'};
4057 end
4058
4059
4060 %% ===== FONTS =====
4061 case 'FigFont'
4062 if ispc
4063 argout1 = 8;
4064 else
4065 argout1 = 9;
4066 end
4067 InterfaceScaling = bst_get('InterfaceScaling');
4068 if (InterfaceScaling ~= 100)
4069 argout1 = argout1 * InterfaceScaling / 100;
4070 end
4071
4072 case 'Font'
4073 % Default font size
4074 if (nargin < 2)
4075 if strncmp(computer,'MAC',3)
4076 fontSize = 12;
4077 else
4078 fontSize = 11;
4079 end
4080 % Font size in input
4081 else
4082 fontSize = varargin{2};
4083 end
4084 % Adjust for interface scaling
4085 fontSize = fontSize * bst_get('InterfaceScaling') / 100;
4086
4087 % Font types
4088 fontTypes = {};
4089 if (nargin >= 3)
4090 if ischar(varargin{3})
4091 fontTypes = varargin(3);
4092 else
4093 fontTypes = varargin{3};
4094 end
4095 else
4096 fontTypes{end + 1} = 'Arial'; % Default font
4097 fontTypes{end + 1} = 'Liberation Sans'; % Free Arial substitute
4098 end
4099 % Check for cached font
4100 foundFont = 0;
4101 for iFont = 1 : length(fontTypes)
4102 strCache = strrep(sprintf('%s%d', fontTypes{iFont}, round(fontSize*100)), ' ', '_');
4103 if ~isempty(GlobalData) && isfield(GlobalData, 'Program') && isfield(GlobalData.Program, 'FontCache') && isfield(GlobalData.Program.FontCache, strCache)
4104 argout1 = GlobalData.Program.FontCache.(strCache);
4105 foundFont = 1;
4106 break;
4107 end
4108 end
4109
4110 % If font not cached, find first supported font
4111 if ~foundFont
4112 ge = java.awt.GraphicsEnvironment.getLocalGraphicsEnvironment();
4113 allFonts = cell(ge.getAvailableFontFamilyNames());
4114
4115 for iFont = 1 : length(fontTypes)
4116 if any(strcmp(fontTypes{iFont}, allFonts))
4117 fontType = fontTypes{iFont};
4118 foundFont = 1;
4119 break;
4120 end
4121 end
4122
4123 if ~foundFont
4124 fontType = 'SansSerif'; % If nothing else works.
4125 end
4126
4127 strCache = strrep(sprintf('%s%d', fontType, round(fontSize*100)), ' ', '_');
4128 argout1 = java.awt.Font(fontType, java.awt.Font.PLAIN, fontSize);
4129 GlobalData.Program.FontCache.(strCache) = argout1;
4130 end
4131
4132 %% ==== PANEL CONTAINERS ====
4133 case 'PanelContainer'
4134 % Get Brainstorm GUI context structure
4135 bst_GUI = GlobalData.Program.GUI;
4136 if (isempty(bst_GUI) || ~isfield(bst_GUI, 'panelContainers'))
4137 error('Brainstorm GUI is not yet initialized');
4138 end
4139
4140 % Get ContainerName in argument
4141 if ((nargin >= 2) && (ischar(varargin{2})))
4142 ContainerName = varargin{2};
4143 % If no container name in argument : just display all the container names
4144 else
4145 disp('Registered panel containers :');
4146 for iContainer = 1:length(bst_GUI.panelContainers)
4147 disp([' - ' bst_GUI.panelContainers(iContainer).name]);
4148 end
4149 return
4150 end
4151
4152 % Look for containerName in all the registered panel containers
4153 iContainer = 1;
4154 found = 0;
4155 while (~found (iContainer <= length(bst_GUI.panelContainers)))
4156 if (strcmpi(ContainerName, bst_GUI.panelContainers(iContainer).name))
4157 found = 1;
4158 else
4159 iContainer = iContainer + 1;
4160 end
4161 end
4162 % If container is found : return it
4163 if (found)
4164 argout1 = bst_GUI.panelContainers(iContainer).jHandle;
4165 else
4166 % warning('Brainstorm:InvalidContainer', 'Container ''%s'' could not be found.', ContainerName);
4167 end
4168
4169
4170 %% ==== PANELS ====
4171 case 'Panel'
4172 % Get Brainstorm GUI context structure
4173 if (isempty(GlobalData) || isempty(GlobalData.Program.GUI) || ~isfield(GlobalData.Program.GUI, 'panels'))
4174 return
4175 end
4176 listPanels = GlobalData.Program.GUI.panels;
4177 % Get Panel in argument
4178 if ((nargin >= 2) && (ischar(varargin{2})))
4179 PanelName = varargin{2};
4180 % If no panel name in argument : just display all the panels names
4181 else
4182 disp('Registered panels :');
4183 for iContainer = 1:length(listPanels)
4184 disp([' - ' get(listPanels(iContainer), 'name')]);
4185 end
4186 return
4187 end
4188 % Look for panelName in all the registered panels
4189 iPanel = find(strcmpi(PanelName, get(listPanels, 'name')), 1);
4190 if ~isempty(iPanel)
4191 argout1 = listPanels(iPanel);
4192 argout2 = iPanel;
4193 end
4194
4195
4196 %% ==== PANEL CONTROLS ====
4197 % Calls : bst_get('PanelControls', PanelName)
4198 case 'PanelControls'
4199 % Get Panel name in argument
4200 if ((nargin >= 2) && (ischar(varargin{2})))
4201 PanelName = varargin{2};
4202 else
4203 error('Invalid call to bst_get()');
4204 end
4205 % Find BstPanel with this name
4206 bstPanel = bst_get('Panel', PanelName);
4207 % If panel was found : return its controls
4208 if ~isempty(bstPanel)
4209 argout1 = get(bstPanel, 'sControls');
4210 end
4211
4212 %% ===== NODES COPY =====
4213 % Calls : bst_get('Clipboard')
4214 case 'Clipboard'
4215 argout1 = GlobalData.Program.Clipboard.Nodes;
4216 argout2 = GlobalData.Program.Clipboard.isCut;
4217
4218 %% ==== DIRECTORIES ====
4219 case 'DirDefaultSubject'
4220 argout1 = '@default_subject';
4221 case 'DirDefaultStudy'
4222 argout1 = '@default_study';
4223 case 'DirAnalysisIntra'
4224 argout1 = '@intra';
4225 case 'DirAnalysisInter'
4226 argout1 = '@inter';
4227 case 'NormalizedSubjectName'
4228 argout1 = 'Group_analysis';
4229
4230 %% ==== OTHER ====
4231 case 'ResizeFunction'
4232 if (bst_get('MatlabVersion') <= 803)
4233 argout1 = 'ResizeFcn';
4234 else
4235 argout1 = 'SizeChangedFcn';
4236 end
4237 case 'groot'
4238 if (bst_get('MatlabVersion') <= 803)
4239 argout1 = 0;
4240 else
4241 argout1 = groot;
4242 end
4243 case 'JFrame'
4244 hFig = varargin{2};
4245 MatlabVersion = bst_get('MatlabVersion');
4246 jFrame = [];
4247 try
4248 if (MatlabVersion <= 705)
4249 jf = get(hFig, 'javaframe');
4250 jFrame = jf.fFigureClient.getWindow();
4251 elseif (MatlabVersion <= 712)
4252 jf = get(handle(hFig), 'javaframe');
4253 jFrame = jf.fFigureClient.getWindow();
4254 elseif (MatlabVersion <= 803)
4255 jf = get(handle(hFig), 'javaframe');
4256 jFrame = jf.fHG1Client.getWindow();
4257 elseif (MatlabVersion < 907) % Matlab >= 2019b deprecated the JavaFrame property
4258 warning('off', 'MATLAB:HandleGraphics:ObsoletedProperty:JavaFrame');
4259 jf = get(hFig, 'javaframe');
4260 warning('on', 'MATLAB:HandleGraphics:ObsoletedProperty:JavaFrame');
4261 jFrame = jf.fHG2Client.getWindow();
4262 else
4263 disp('BST> Error: Matlab 2019b deprecated the JavaFrame property.');
4264 end
4265 catch
4266 disp('BST> Warning: Cannot get the JavaFrame property for the selected figure.');
4267 end
4268 argout1 = jFrame;
4269
4270 %% ==== ERROR ====
4271 otherwise
4272 error(sprintf('Invalid context : "%s"', contextName));
4273 end
4274 end
4275
4276
4277
4278
4279 %% ==== HELPERS ====
4280 % Return all the protocol studies that have a given file in its structures
4281 % Possible field names: Result.DataFile, Result.FileName, Data.FileName, Channel.FileName
4282 %
4283 % USAGE: [sFoundStudy, iFoundStudy, iItem] = findFileInStudies(fieldGroup, fieldName, fieldFile, iStudiesList)
4284 % [sFoundStudy, iFoundStudy, iItem] = findFileInStudies(fieldGroup, fieldName, fieldFile)
4285 function [sFoundStudy, iFoundStudy, iItem] = findFileInStudies(fieldGroup, fieldName, fieldFile, iStudiesList)
4286 global GlobalData;
4287 sFoundStudy = [];
4288 iFoundStudy = [];
4289 iItem = [];
4290 % If no file provided, return
4291 if isempty(fieldFile)
4292 return;
4293 end
4294 % Extract folder(s) of the file(s) we're looking for
4295 fieldParts = strfind(fieldFile, '|');
4296 if ~isempty(fieldParts)
4297 fieldParts(end+1) = length(fieldFile);
4298 fieldFolders = {};
4299 iLast = 1;
4300 for iPart = 1:length(fieldParts)
4301 folder = fileparts(fieldFile(iLast:fieldParts(iPart)-1));
4302 if ~isempty(folder)
4303 fieldFolders{end + 1} = folder;
4304 end
4305 iLast = fieldParts(iPart) + 1;
4306 end
4307 else
4308 fieldFolders = {fileparts(fieldFile)};
4309 end
4310 % Get protocol information
4311 ProtocolStudies = GlobalData.DataBase.ProtocolStudies(GlobalData.DataBase.iProtocol);
4312 % List studies to process
4313 if (nargin < 4) || isempty(iStudiesList)
4314 iStudiesList = [-2, -3, 1:length(ProtocolStudies.Study)];
4315 end
4316
4317 % NORMAL STUDIES: Look for surface file in all the surfaces of all subjects
4318 for iStudy = iStudiesList
4319 % Get study
4320 switch (iStudy)
4321 case -2, sStudy = ProtocolStudies.AnalysisStudy;
4322 case -3, sStudy = ProtocolStudies.DefaultStudy;
4323 otherwise, sStudy = ProtocolStudies.Study(iStudy);
4324 end
4325 % Check if field is available for the study
4326 if isempty(sStudy.(fieldGroup))
4327 continue;
4328 end
4329 % Check we are in the correct folder
4330 if ~any(file_compare(fieldFolders, fileparts(sStudy.FileName)))
4331 continue;
4332 end
4333 % Get list of files from study
4334 filesList = {sStudy.(fieldGroup).(fieldName)};
4335 if isempty(filesList)
4336 continue;
4337 end
4338 % Replace empty cells with empty strings
4339 iValidFiles = find(cellfun(@ischar, filesList));
4340 if isempty(iValidFiles)
4341 continue;
4342 end
4343 % Find target in this list
4344 iItem = find(file_compare(filesList(iValidFiles), fieldFile));
4345 if ~isempty(iItem)
4346 sFoundStudy = sStudy;
4347 iFoundStudy = iStudy;
4348 iItem = iValidFiles(iItem);
4349 return
4350 end
4351 end
4352 end
4353
4354
4355 %% ===== FILL MISSING FIELDS =====
4356 function bstPref = FillMissingFields(PrefName, defPref)
4357 global GlobalData;
4358 if isfield(GlobalData, 'Preferences') && isfield(GlobalData.Preferences, PrefName) && isstruct(GlobalData.Preferences.(PrefName))
4359 bstPref = GlobalData.Preferences.(PrefName);
4360 bstPref = struct_copy_fields(bstPref, defPref, 0);
4361 else
4362 bstPref = defPref;
4363 end
4364 end
4365
4366
4367
4368
4369
4370
4371
To change parameters or database structures: bst_set.m.
1 function bst_set( varargin )
2 % BST_SET: Set a Brainstorm structure.
3 %
4 % DESCRIPTION: This function is used to abstract the way that these structures are stored.
5 %
6 % USAGE:
7 % ====== DIRECTORIES ==================================================================
8 % - bst_set('BrainstormHomeDir', BrainstormHomeDir)
9 % - bst_set('BrainstormTmpDir', BrainstormTmpDir)
10 % - bst_set('BrainstormDbDir', BrainstormDbDir)
11 % - bst_set('LastUsedDirs', sDirectories)
12 % - bst_set('BrainSuiteDir', BrainSuiteDir)
13 % - bst_set('PythonExe', PythonExe)
14 % - bst_set('PluginCustomPath', PluginCustomPath)
15 %
16 % ====== PROTOCOLS ====================================================================
17 % - bst_set('iProtocol', iProtocol)
18 % - bst_set('ProtocolInfo', sProtocolInfo)
19 % - bst_set('ProtocolSubjects', ProtocolSubjects)
20 % - bst_set('isProtocolLoaded', isProtocolLoaded)
21 % - bst_set('isProtocolModified',isProtocolModified)
22 % - bst_set('ProtocolStudies', ProtocolStudies)
23 % - bst_set('Study', iStudy, sStudy) : Set a study in current protocol
24 % - bst_set('Subject', iSubject, sSubject) : Set a subject in current protocol
25 %
26 % ====== GUI =================================================================
27 % - bst_set('Layout', sLayout)
28 % - bst_set('Layout', PropName, PropValue)
29 % - bst_set('Clipboard', Nodes, isCut) : Copy operation from the tree
30 %
31 % ====== CONFIGURATION =================================================================
32 % - bst_set('Version', Version)
33 % - bst_set('ByteOrder', value) : 'b' for big endian, 'l' for little endian
34 % - bst_set('AutoUpdates', isAutoUpdates)
35 % - bst_set('ExpertMode', isExpertMode)
36 % - bst_set('DisplayGFP', isDisplayGFP)
37 % - bst_set('DownsampleTimeSeries', isDownsampleTimeSeries)
38 % - bst_set('GraphicsSmoothing', isGraphicsSmoothing)
39 % - bst_set('ForceMatCompression', isForceCompression)
40 % - bst_set('IgnoreMemoryWarnings', isIgnoreMemoryWarnings)
41 % - bst_set('SystemCopy', isSystemCopy)
42 % - bst_set('DisableOpenGL', isDisableOpenGL)
43 % - bst_set('InterfaceScaling', InterfaceScaling)
44 % - bst_set('TSDisplayMode', TSDisplayMode) : {'butterfly','column'}
45 % - bst_set('ElectrodeConfig', ElectrodeConfig, Modality)
46 % - bst_set('ElecInterpDist', ElecInterpDist, Modality)
47 % - bst_set('DefaultFormats' defaultFormats)
48 % - bst_set('BFSProperties', [scalpCond,skullCond,brainCond,scalpThick,skullThick])
49 % - bst_set('ImportEegRawOptions', ImportEegRawOptions)
50 % - bst_set('BugReportOptions', BugReportOptions)
51 % - bst_set('DefaultSurfaceDisplay', displayStruct)
52 % - bst_set('MagneticExtrapOptions', extrapStruct)
53 % - bst_set('TimefreqOptions_morlet', Options)
54 % - bst_set('TimefreqOptions_fft', Options)
55 % - bst_set('TimefreqOptions_psd', Options)
56 % - bst_set('TimefreqOptions_hilbert', Options)
57 % - bst_set('TimefreqOptions_plv', Options)
58 % - bst_set('OpenMEEGOptions', Options)
59 % - bst_set('DuneuroOptions', Options)
60 % - bst_set('GridOptions_headmodel', Options)
61 % - bst_set('GridOptions_dipfit', Options)
62 % - bst_set('UniformizeTimeSeriesScales', isUniform)
63 % - bst_set('FlipYAxis', isFlipY)
64 % - bst_set('AutoScaleY', isAutoScaleY)
65 % - bst_set('FixedScaleY', Modality, Value)
66 % - bst_set('XScale', XScale)
67 % - bst_set('YScale', YScale)
68 % - bst_set('ShowXGrid', isShowXGrid)
69 % - bst_set('ShowYGrid', isShowYGrid)
70 % - bst_set('ShowZeroLines', isShowZeroLines)
71 % - bst_set('ShowEventsMode', ShowEventsMode)
72 % - bst_set('Resolution', [resX,resY])
73 % - bst_set('UseSigProcToolbox', UseSigProcToolbox)
74 % - bst_set('RawViewerOptions', RawViewerOptions)
75 % - bst_set('TopoLayoutOptions', TopoLayoutOptions)
76 % - bst_set('StatThreshOptions', StatThreshOptions)
77 % - bst_set('ContactSheetOptions', ContactSheetOptions)
78 % - bst_set('ProcessOptions', ProcessOptions)
79 % - bst_set('MriOptions', MriOptions)
80 % - bst_set('CustomColormaps', CustomColormaps)
81 % - bst_set('DigitizeOptions', DigitizeOptions)
82 % - bst_set('PcaOptions', PcaOptions)
83 % - bst_set('ReadOnly', ReadOnly)
84 % - bst_set('LastPsdDisplayFunction', LastPsdDisplayFunction)
85 % - bst_set('LastTfDisplayFunction', LastTfDisplayFunction)
86 % - bst_set('PlotlyCredentials', Username, ApiKey, Domain)
87 % - bst_set('KlustersExecutable', ExecutablePath)
88 % - bst_set('ExportBidsOptions'), ExportBidsOptions)
89 % - bst_set('Pipelines') Saved Pipelines stored
90 %
91 % SEE ALSO bst_get
92
93 % @=============================================================================
94 % This function is part of the Brainstorm software:
95 % https://neuroimage.usc.edu/brainstorm
96 %
97 % Copyright (c) University of Southern California & McGill University
98 % This software is distributed under the terms of the GNU General Public License
99 % as published by the Free Software Foundation. Further details on the GPLv3
100 % license can be found at http://www.gnu.org/copyleft/gpl.html.
101 %
102 % FOR RESEARCH PURPOSES ONLY. THE SOFTWARE IS PROVIDED "AS IS," AND THE
103 % UNIVERSITY OF SOUTHERN CALIFORNIA AND ITS COLLABORATORS DO NOT MAKE ANY
104 % WARRANTY, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO WARRANTIES OF
105 % MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE, NOR DO THEY ASSUME ANY
106 % LIABILITY OR RESPONSIBILITY FOR THE USE OF THIS SOFTWARE.
107 %
108 % For more information type "brainstorm license" at command prompt.
109 % =============================================================================@
110 %
111 % Authors: Francois Tadel, 2008-2021
112 % Martin Cousineau, 2017
113
114 global GlobalData;
115
116 %% ==== PARSE INPUTS ====
117 if ((nargin >= 1) && ischar(varargin{1}))
118 contextName = varargin{1};
119 if (nargin >= 2)
120 contextValue = varargin{2};
121 else
122 contextValue = [];
123 end
124 else
125 error('Usage : bst_set(contextName, contextValue)');
126 end
127
128 % Get required context structure
129 switch contextName
130 %% ==== BRAINSTORM CONFIGURATION ====
131 case 'Version'
132 GlobalData.Program.Version = contextValue;
133 case 'BrainstormHomeDir'
134 GlobalData.Program.BrainstormHomeDir = contextValue;
135 case 'BrainstormDbDir'
136 GlobalData.DataBase.BrainstormDbDir = contextValue;
137 case 'BrainstormTmpDir'
138 GlobalData.Preferences.BrainstormTmpDir = contextValue;
139 case 'Pipelines'
140 GlobalData.Processes.Pipelines = contextValue;
141
142 %% ==== PROTOCOL ====
143 case 'iProtocol'
144 if isnumeric(contextValue)
145 GlobalData.DataBase.iProtocol = contextValue;
146 else
147 error('iProtocol should be a number.');
148 end
149 case {'ProtocolSubjects', 'ProtocolStudies'}
150 for structField = fieldnames(contextValue)'
151 GlobalData.DataBase.(contextName)(GlobalData.DataBase.iProtocol).(structField{1}) = contextValue.(structField{1});
152 end
153 GlobalData.DataBase.isProtocolModified(GlobalData.DataBase.iProtocol) = 1;
154 case 'ProtocolInfo'
155 for structField = fieldnames(contextValue)'
156 GlobalData.DataBase.(contextName)(GlobalData.DataBase.iProtocol).(structField{1}) = contextValue.(structField{1});
157 end
158 case 'isProtocolLoaded'
159 GlobalData.DataBase.isProtocolLoaded(GlobalData.DataBase.iProtocol) = contextValue;
160 case 'isProtocolModified'
161 GlobalData.DataBase.isProtocolModified(GlobalData.DataBase.iProtocol) = contextValue;
162
163 %% ==== SUBJECT ====
164 case 'Subject'
165 % Get subjects list
166 ProtocolSubjects = bst_get('ProtocolSubjects');
167 iSubject = varargin{2};
168 sSubject = varargin{3};
169 % If default subject
170 if (iSubject == 0)
171 ProtocolSubjects.DefaultSubject = sSubject;
172 else
173 ProtocolSubjects.Subject(iSubject) = sSubject;
174 end
175 % Update DataBase
176 bst_set('ProtocolSubjects', ProtocolSubjects);
177
178
179 %% ==== STUDY ====
180 case 'Study'
181 % Get studies list
182 ProtocolStudies = bst_get('ProtocolStudies');
183 iStudies = varargin{2};
184 sStudies = varargin{3};
185 iAnalysisStudy = -2;
186 iDefaultStudy = -3;
187 for i = 1:length(iStudies)
188 % Normal study
189 if (iStudies(i) > 0)
190 ProtocolStudies.Study(iStudies(i)) = sStudies(i);
191 % Inter-subject analysis study
192 elseif (iStudies(i) == iAnalysisStudy)
193 ProtocolStudies.AnalysisStudy = sStudies(i);
194 % Default study
195 elseif (iStudies(i) == iDefaultStudy)
196 ProtocolStudies.DefaultStudy = sStudies(i);
197 end
198 end
199 % Update DataBase
200 bst_set('ProtocolStudies', ProtocolStudies);
201
202
203 %% ==== GUI ====
204 % USAGE: bst_set('Layout', sLayout)
205 % bst_set('Layout', PropName, PropValue)
206 case 'Layout'
207 if (nargin == 2) && isstruct(contextValue)
208 GlobalData.Preferences.Layout = contextValue;
209 isUpdateScreens = 0;
210 elseif (nargin == 3) && ischar(contextValue) && isfield(GlobalData.Preferences, 'Layout') && isfield(GlobalData.Preferences.Layout, contextValue)
211 GlobalData.Preferences.Layout.(contextValue) = varargin{3};
212 isUpdateScreens = strcmpi(contextValue, 'DoubleScreen');
213 else
214 error('Invalid call to bst_set.');
215 end
216 % Update screen configuration
217 GlobalData.Program.ScreenDef = gui_layout('GetScreenClientArea');
218 % Update layout right now
219 gui_layout('Update');
220 % If the number of screen was changed: update the maximum size of the Brainstorm window
221 if isUpdateScreens
222 gui_layout('UpdateMaxBstSize');
223 end
224
225 % USAGE: bst_set('FixedScaleY', [])
226 % bst_set('FixedScaleY', Modality, Value)
227 case 'FixedScaleY'
228 if (nargin == 3) && ~isempty(contextValue) && ~isempty(varargin{3})
229 GlobalData.Preferences.FixedScaleY.(contextValue) = varargin{3};
230 elseif (nargin == 2) && isempty(contextValue)
231 GlobalData.Preferences.FixedScaleY = struct();
232 end
233
234 case 'ByteOrder'
235 switch(contextValue)
236 case {'b','ieee-le','n'}
237 GlobalData.Preferences.ByteOrder = 'b';
238 case {'l','ieee-be'}
239 GlobalData.Preferences.ByteOrder = 'l';
240 otherwise
241 error('Invalid byte order.');
242 end
243
244 case 'Clipboard'
245 if (length(varargin) >= 3)
246 isCut = varargin{3};
247 else
248 isCut = 0;
249 end
250 GlobalData.Program.Clipboard.Nodes = contextValue;
251 GlobalData.Program.Clipboard.isCut = isCut;
252
253 case 'ElectrodeConfig'
254 Modality = varargin{2};
255 ElectrodeConf = varargin{3};
256 if isequal(Modality, 'ECOG+SEEG')
257 Modality = 'ECOG_SEEG';
258 elseif ~ismember(Modality, {'EEG','SEEG','ECOG','MEG'})
259 error(['Invalid modality: ' Modality]);
260 end
261 GlobalData.Preferences.(contextName).(Modality) = ElectrodeConf;
262
263 case 'ElecInterpDist'
264 Modality = varargin{2};
265 ElecInterpDist = varargin{3};
266 if isequal(Modality, 'ECOG+SEEG')
267 Modality = 'ECOG_SEEG';
268 elseif ~ismember(Modality, {'EEG','SEEG','ECOG','MEG'})
269 error(['Invalid modality: ' Modality]);
270 end
271 GlobalData.Preferences.(contextName).(Modality) = ElecInterpDist;
272
273 case {'UniformizeTimeSeriesScales', 'XScale', 'YScale', 'FlipYAxis', 'AutoScaleY', 'ShowXGrid', 'ShowYGrid', 'ShowZeroLines', 'ShowEventsMode', ...
274 'Resolution', 'AutoUpdates', 'ExpertMode', 'DisplayGFP', 'ForceMatCompression', 'GraphicsSmoothing', 'DownsampleTimeSeries', 'UseCrossPlatformJLF', 'ShowProcessTooltip' ...
275 'DisableOpenGL', 'InterfaceScaling', 'TSDisplayMode', 'UseSigProcToolbox', 'LastUsedDirs', 'DefaultFormats', ...
276 'BFSProperties', 'ImportDataOptions', 'ImportEegRawOptions', 'RawViewerOptions', 'MontageOptions', 'TopoLayoutOptions', ...
277 'StatThreshOptions', 'ContactSheetOptions', 'ProcessOptions', 'BugReportOptions', 'DefaultSurfaceDisplay', ...
278 'MagneticExtrapOptions', 'MriOptions', 'ConnectGraphOptions', 'NodelistOptions', 'IgnoreMemoryWarnings', 'SystemCopy', ...
279 'TimefreqOptions_morlet', 'TimefreqOptions_hilbert', 'TimefreqOptions_fft', 'TimefreqOptions_psd', 'TimefreqOptions_stft', 'TimefreqOptions_plv', ...
280 'OpenMEEGOptions', 'DuneuroOptions','NIRSTORMOptions', 'DigitizeOptions', 'PcaOptions', 'CustomColormaps', 'PluginCustomPath', 'BrainSuiteDir', 'PythonExe', ...
281 'GridOptions_headmodel', 'GridOptions_dipfit', 'LastPsdDisplayFunction', 'LastTfDisplayFunction', 'KlustersExecutable', 'ExportBidsOptions', 'ShowHiddenFiles'}
282 GlobalData.Preferences.(contextName) = contextValue;
283
284 case 'ReadOnly'
285 GlobalData.DataBase.isReadOnly = contextValue;
286
287 case 'PlotlyCredentials'
288 if length(varargin) ~= 4
289 error('Invalid call to bst_set.');
290 end
291 [username, apiKey, domain] = varargin{2:4};
292 % Default domain: plot.ly
293 if isempty(domain)
294 domain = 'https://plot.ly';
295 end
296 % Plotly needs a URL with HTTP and no trailing slash.
297 if strfind(domain, 'https://')
298 domain = strrep(domain, 'https://', 'http://');
299 elseif isempty(strfind(domain, 'http://'))
300 domain = ['http://', domain];
301 end
302 if domain(end) == '/'
303 domain = domain(1:end-1);
304 end
305 % Save credentials
306 saveplotlycredentials(username, apiKey);
307 saveplotlyconfig(domain);
308
309 %% ==== ERROR ====
310 otherwise
311 error('Invalid context : ''%s''', contextName);
312
313
314 end
315
316
317
318
File structures
The structures of the different types of files functions were described in the sections "On the hard drive" of the introduction tutorials. Here is a directory of all these sections:
Continuous recordings: File type "raw", Event markers.
Scouts and atlases: Saved in the surface files, Scouts.
Custom processing
In many situations, you will find useful to read the files available in the database, and maybe modify them. The easiest approaches do not require any scripting, we will start by reviewing them quickly.
Process: Run Matlab command
If you want to modify the values saved in a file (eg. the field "F" from a "data" file), the easiest way is probably to use the process1 File > Run Matlab command. It is also available from Process2 in the category "Other".
It loads the files in input and run them through a piece of Matlab code that you can edit freely. It can extend a lot the flexibility of the Brainstorm pipeline manager, providing an easy access to any Matlab function or script.
DataMat is the exact content of the corresponding .mat file, as loaded with Matlab's load() function. Edit some of the fields of this structure from the Matlab command window:
Now right-click on the folder containing the original file > File > Import from Matlab > DataMat:
If instead, you right-click on the original file and select the menu File > Import from Matlab, it overwrites the selected file instead of creating a new one with the selected structure.
Reference: File manipulation
Useful functions for manipulating file names and paths (read the code of the functions for help):
file_fullpath: Converts a relative file path from the Brainstorm database to an absolute path.
file_short: Converts an absolute file path to a short path, relative to the current protocol folder.
file_gettype: Returns the type of a file.
Reading files from a script (all the functions take relative paths in input):
in_bst_data(DataFile): Read an imported epoch.
in_bst_timefreq(TimefreqFile): Read a power spectrum, time-frequency or connectivity file.
in_bst_headmodel(HeadmodelFile, ApplyOrient): Read a head model file and apply orientations.
in_bst_results(ResultsFile, LoadFull): Load a source file and optionally reconstruct the full source time series on the fly (ImagingKernel * recordings).
in_bst_matrix(MatrixFile): Read a file with the type "matrix".
in_bst(FileName, TimeWindow): Read any Brainstorm data file with the possibility to load only a specific part of the file. "TimeWindow" is a range of time values in seconds: [tStart, tStop].
bst_process('LoadInputFile', FileName, Target, TimeWindow): The most high-level function for reading data files. "Target" is a string with the list of signal names or types to load.
bst_memory('GetConnectMatrix', TimefreqMat): Rebuild a full connectivity matrix.
in_fopen(DataFile, FileFormat, ImportOptions): Low-level function for opening continuous files.
in_fread(sFile, ChannelMat, iEpoch, SamplesBounds, iChannels, ImportOptions): Low-level function for reading blocks from continuous files. Require a call to in_fopen first to get the sFile structure.
Saving files:
bst_save(FileName, FileMat, Version, isAppend): Save a file but does not register it in the database.
FileName: Must be an absolute path, use in combination with file_fullpath to use relative paths.
FileMat: Must be a valid Brainstorm structure, corresponding to the file type.
Version: Defines which version of the Matlab .mat format is used to store the data:
'v6': Fastest option, bigger files, no compression, no files >2Gb
'v7': Slower option, compressed, no files >2Gb
'v7.3': Much slower than the others, compressed, but only way to save files > 2Gb.
isAppend: If set to 1, updates only the fields defined in FileMat, keep the others untouched.
file_unique: Produces a unique file name by adding a number tag to it.
bst_process('GetNewFilename', OutputFolder, BaseFilename): Generate a new unique file name based on the beginning of the file name (eg. BaseFilename='data_average_test_').
Registering new files in the database:
db_add(iStudy/iSubject, FileMat): Add a new file in an anatomy folder or a functional data folder. This function saves and then registers it in the database. FileMat must be a structure, not a filename. You should not save the file manually before calling this function.
db_add_data(iStudy, FileName, FileMat): Register in the database a structure FileMat that has already been saved in file FileName. You should call bst_save manually before calling this function.
db_add_data(iStudy, FileName, FileMat, iItem): Overwrites the existing file #iItem.
Reload folders (if you saved or deleted files without registering correctly the modification in the database):
db_reload_studies(iStudies): Reload only the select data folders (aka "studies").
db_reload_conditions(iSubjects): Reload all the data folders for a subject.
db_reload_subjects(iSubjects): Reload the anatomy of the selected subjects.
db_reload_database('current'): Reload the entire protocol (anatomy and functional data).
Other useful database functions:
db_add_condition: Create a new folder in a subject.
db_add_subject: Create a new subject.
db_delete_studies: Delete a list of folders.
db_delete_subjects: Delete a list of subjects.
db_group_conditions: Merge two analysis folders from the same subject.
db_rename_condition: Rename a folder.
db_rename_subject: Rename a subject
db_set_channel: Set the channel file for a folder.
db_set_noisecov: Set the noise/data covariance for a folder, or copy to other folders/subjects.
db_set_template: Copy an anatomy template to a subject or use it as the default anatomy.
Export a file from the database to other file formats (read the comments in the functions for help):
export_channel
export_data
export_events
export_result
export_timefreq
export_matrix
export_mri
export_surfaces
export_protocol: Export a subject or an entire protocol as a .zip file.
Convert Brainstorm structures to FieldTrip structures:
out_fieldtrip_channel
out_fieldtrip_data
out_fieldtrip_timefreq
out_fieldtrip_headmodel
out_fieldtrip_results
out_fieldtrip_matrix
Reference: Display functions [TODO]
Plot file contents:
view_channels: Display sensors in a 3D figure.
view_helmet: Display the inner surface of the MEG helmet in a 3D figure.
view_timeseries: Display a data file as time series.
view_timeseries_matrix: Display a custom matrix as time series.
view_topography: Display a data file as a spatial topography.
view_erpimage: Display multiple data files as an image, signal by signal.
view_timefreq: Open a time-frequency file (various display modes available).
view_spectrum: Display power spectrum (PSD or time-frequency files).
view_connect: Open a connectivity matrix (various display modes available).
view_matrix: Open a "matrix" file (various display modes available).
view_contactsheet: Create a contact sheet in time or across a volume from an existing figure.
view_noisecov: Display a noise or data covariance file.
view_dipoles: Open a dipoles file (various display modes available).
view_pac: Open PAC results (various display modes available).
view_mri: View a MRI file in the MRI viewer (with or without a functional overlay from a source file).
view_mri_3d: View a MRI file in a 3D figure (with or without a functional overlay from a source file).
view_surface: View a surface.
view_surface_data: View a surface file with a source file as its texture.
bst_colormaps('SetMaxMode', ColormapType, MaxMode): Colorbar range ('global', 'local', 'custom').
bst_colormaps('SetMaxCustom', ColormapType, [], Min, Max): Set a custom colorbar range.
bst_colormaps('SetColormapAbsolute', ColormapType, isAbsolute): Show positive or relative values
bst_colormaps('SetDisplayColorbar', ColormapType, isDisplay): Show colorbar in the figures.
bst_colormaps('RestoreDefaults', ColormapType): Restore the default configuration for a colormap.
ColormapType: anatomy, meg, eeg, sources, stat1, stat2, time, timefreq, connect1, connectn, image
Configure statistical thresholding:
StatThreshOptions = bst_get('StatThreshOptions'): Get display options selected in the Stat tab.
bst_set('StatThreshOptions', StatThreshOptions): Change the options selected in the Stat tab.
StatThreshOptions.pThreshold: Current p-Value threshold.
StatThreshOptions.Correction: Correction for multiple comparisons ('none', 'fdr', 'bonferroni')
StatThreshOptions.Control: List of dimensions to correct for multiple comparisons (default = [1 2 3])
Export the contents of a figure to a file:
out_figure_image: Screen capture of any Brainstorm figure.
out_figure_movie: Save a movie from one or multitple Brainstorm figures.
out_figure_timefreq: Extract some of the data displayed in a time-frequency figure.
out_figure_timeseries: Extract some of the data displayed in a time series figure.
Example: Creating a new file [TODO]
(explain the reloading)
Example: Edit events [TODO]
A step that commonly requires manual changes is the definition of the event markers. Many time we have to integrate external triggers, behavioral responses ,etc
The easier way to understand how to use a function is to search the code with the "Find files" interface in Matlab. Go to the brainstorm3 folder, click on "File files" (or Ctrl+Shift+F), enter the name of function your are looking for in "Find files containing text", Include subfolders, Match case. It will return all the lines that include the string you entered across all the files in Brainstorm. Just double-click on a line to jump to the code in the Matlab editor.
You can use the same interface to find what function is called when you click on a button or menu in the interface. Search for the label or the tooltip of the interface element in the same way. The example below shows how to track what happens when you click on the headmodel popup menu "Check spheres".
If you have trouble understanding how to set some input parameters, you can use the debugger to explore a real use case. Place a breakpoint at the begging of your function of interest (watch this tutorial if you don't know how to do it), for in example view_timeseries.m. Then click on the corresponding menus in the Brainstorm interface (eg. double-click on on a data file). When the execution reaches the line you selected, it stops and give you back the commands. You can explore the values in all the variables, modify them, and execute the code step by step (many options available in the Editor tab of Matlab).
Additional quality control [TODO]
You cann add in the reports all the information that will help you control the quality of the analysis, or even figures you want to include in publications or clinical reports.
Creating loops is not supported yet by the script generator, but relatively easy to do from a script without having to know too much about Matlab programming.
1) Fill the cell array SubjectNames with all your subjects names, with the same dimensions as the list of input raw files (sFiles)
2) Add a "for" loop that includes all the bst_process() calls (leave the bst_report() calls and input definition outside)
3) Inside the loop, replace SubjectNames with SubjectNames{i} and sFiles with sFiles(i)
How to process an entire study
This section proposes a standard workflow for processing a full group study with Brainstorm. It contains the same steps of analysis as the introduction tutorials, but separating what can be done automatically from what should be done manually. This workflow can be adapted to most ERP studies (stimulus-based).
Prototype: Start by processing one or two subjects completely interactively (exactly like in the introduction tutorials). Use the few pilot subjects that you have for your study to prototype the analysis pipeline and check manually all the intermediate stages. Take notes of what you're doing along the way, so that you can later write a script that reproduces the same operations.
Anatomical fiducials: Set NAS/LPA/RPA and compute the MNI transformation for each subject.
Segmentation: Run FreeSurfer/BrainSuite to get surfaces and atlases for all the subjects.
File > Batch MRI fiducials: This menu prompts for the selection of the fiducials for all the subjects and saves a file fiducials.m in each segmentation folder. You will not have to redo this even if you have to start over your analysis from the beginning.
Script: Write a loop that calls the process "Import anatomy folder" for all the subjects.
Alternatives: Create and import the subjects one by one and set the fiducials at the import time. Or use the default anatomy for all the subjects (or use warped templates).
Script #1: Pre-processing: Loop on the subjects and the acquisition runs.
Create link to raw files: Link the subject and noise recordings to the database.
Event markers: Read and group triggers from digital and analog channel, fix stimulation delays
Evaluation: Power spectrum density of the recordings to evaluate their quality.
Compute SSP: Heartbeats, Blinks (this selects the first component of each decomposition)
Compute ICA: If you have some artifacts you'd like to remove with ICA (no default selection).
Screenshots: Check the MRI/sensors registration, PSD before and after corrections, SSP.
Export the report: One report per subject, or one report for all the subjects, saved in HTML.
Manual inspection #1:
Check the reports: Information messages (number of events, errors and warnings) and screen captures (registration problems, obvious noisy channels, incorrect SSP topographies).
Mark bad channels: Open the recordings, select the channels and mark them as bad. Or use the process "Set bad channels" to mark the same bad channels in multiple files.
Fix the SSP/ICA: For the suspicious runs: Open the file, adjust the list of blink and cardiac events, remove and recompute the SSP decompositions, manually select the components.
Detect other artifacts: Run the process on all the runs of all the subjects at once (select all the files in Process1 and run the process, or generate the equivalent script).
Mark bad segments: Review the artifacts detected in 1-7Hz and 40-240Hz, keep only the ones you really want to remove, then mark the event categories as bad. Review quickly the rest of the file and check that there are no other important artifacts.
Additional SSP: If you find one type of artifact that repeats (typically saccades and SQUID jumps), you can create additional SSP projectors, either with the process "SSP: Generic" or directly from a topography figure (right-click on the figure > Snapshot> Use as SSP projector).
Importing: Process "Import MEG/EEG: Events" and "Pre-process > Remove DC offset".
Averaging: Average trials by run, average runs by subject (registration problem in MEG).
Noise covariance: Compute from empty room or resting recordings, copy to other folders.
Head model: Compute for each run, or compute once and copy if the runs are co-registered.
Sources: Compute for each run, average across runs and subjects in source space for MEG.
Time-frequency: Computation with Hilbert transform or Morlet wavelets, then normalize.
Screenshots: Check the quality of all the averages (time series, topographies, sources).
Export the report: One report per subject, or one report for all the subjects, saved in HTML.
Manual inspection #2:
Check the reports: Check the number of epochs imported and averaged in each condition, check the screen capture of the averages (all the primary responses should be clearly visible).
Regions of interest: If not using predefined regions from an atlas, define the scouts on the anatomy of each subject (or on the template and then project them to the subjects).
Script #3: Group analysis, ROI-based analysis, etc.
Averaging: Group averages for the sensor data, the sources and the time-frequency maps.
Statistics: Contrast between conditions or groups of subjects.
Regions of interest: Any operation that involve scouts.
Final script
The following script from the Brainstorm distribution reproduces the introduction tutorials ("Get started"): brainstorm3/toolbox/script/tutorial_introduction.m - Report: report_TutorialIntroduction.html