gusucode.com > 信号处理工具箱 - signal源码程序 > signal\signal\siggui\sptool.m

    function varargout = sptool(varargin)
%SPTOOL  Signal Processing Tool - Graphical User Interface.
%   SPTOOL opens the SPTool window which allows you to import,
%   analyze, and manipulate signals, filters, and spectra.
%   
%   Exporting Component Structures from SPTool via the Command Line
%   ---------------------------------------------------------------
%   The following commands export component structures from the
%   currently open SPTool.
%
%   s = sptool('Signals') returns a structure array of all the signals.
%   f = sptool('Filters') returns a structure array of all the filters.
%   s = sptool('Spectra') returns a structure array of all the spectra.
%
%   [s,ind] = sptool(...) returns an index vector indicating which
%   of the elements of s are currently selected in SPTool.
%
%   s = sptool(...,0) returns only the currently selected objects.
%
%   Creating and Loading Component Structures via the Command Line
%   --------------------------------------------------------------
%   struc = sptool('create',PARAMLIST) creates a component
%   structure, struc, (defined by PARAMLIST) in the workspace.
%
%   sptool('load',struc) loads struc into SPTool; opens SPTool if
%   necessary.
%
%   struc = sptool('load',PARAMLIST) loads the component structure
%   defined by PARAMLIST into SPTool; if an optional output argument is
%   specified, a component structure, struc, is created in the workspace.
%
%   COMPONENT           PARAMLIST
%   ~~~~~~~~~           ~~~~~~~~~
%   SIGNALS:   COMPONENT_NAME,DATA,FS,LABEL
%   FILTERS:   COMPONENT_NAME,NUM,DEN,FS,LABEL
%   SPECTRA:   COMPONENT_NAME,DATA,F,LABEL
%
%   PARAMETER DEFINITIONS
%   ~~~~~~~~~~~~~~~~~~~~~
%   COMPONENT_NAME - 'Signal', 'Filter', or 'Spectrum'; if omitted
%             COMPONENT_NAME defaults to 'Signal'.
%   DATA    - a vector of doubles representing a signal or a spectrum.
%   NUM,DEN - numerator and denominator coefficients of the filter's
%             response in transfer function form.
%   FS      - sampling frequency (OPTIONAL), defaults to 1.
%   F       - frequency vector; applies to spectrum components only.
%   LABEL   - a string specifying the variable name of the component
%             as it will appear in SPTool (OPTIONAL); defaults to one
%             of the following: 'sig', 'filt', or 'spec'.

%  API for calling other tools - 
%     Components and clients are defined by 'components' data
%     structure in the file signal/private/sptcompp.m.  See this
%     file for details.
%     Verb Button Pressed - calls 
%         feval(verb.owningClient,'action',verb.action)
%     Selection has changed - calls for ALL verbs
%         enable = feval(verb.owningClient,'selection',verb.action,msg)
%         and sets verb's enable to result
%     Closing SPTool - calls
%         feval(defaultClient,'SPTclose')

%   Copyright (c) 1988-98 by The MathWorks, Inc.
% $Revision: 1.3 $

% Authors: T. Krauss and B. Jones

if nargin == 0
    action = 'init';
    shh = get(0,'ShowHiddenHandles');
    set(0,'ShowHiddenHandles','on')
    spfig = findobj(0,'Tag','sptool');
    set(0,'ShowHiddenHandles',shh);
    if ~isempty(spfig)
        figure(spfig)
        return
    end
else
    action = varargin{1};
end

switch action

case 'init'                % initialization
    
    % Initialize SPTool userdata (ud)
    ud = init_sptool_data_settings;
    
    % create sptool manager gui
    [fig,ud] = create_sptool_gui(ud);
    
    set(fig,'userdata',ud,'resizefcn','sptool(''resize'')')
    
    allPanels = {ud.prefs.panelName};
    defsesInd = findcstr(allPanels,'defsession');
    fileExist = ~isempty(which('startup.spt'));
    if ud.prefs(defsesInd).currentValue{1}(1) & fileExist,
        % Load default SPTool session file
        sptool('open','startup.spt');
    end

    sptool('resize')
    selectionChanged(fig,'new')
    set(fig,'HandleVisibility','callback','visible','on')
        
%------------------------------------------------------------------------
% structArray = sptool('callall',fname,structArray)
% searches for all fname.m on path
% and calls all found with structArray = feval(fname,structArray)
case 'callall'
    fname = varargin{2};
    structArray = varargin{3};
    
    w = which('-all',fname);
    % make sure each entry is unique
    for i=length(w):-1:1
        ind = findcstr(w(1:i-1),w{i});
        if ~isempty(ind)
            w(i) = [];
        end
    end
    if length(w)>0
        origPath=pwd;
    end
    for i=1:length(w)
        thispath=char(w(i));
        pathsp=find(thispath==filesep);
        if isvms
            pathsp=find(thispath==']');
        end
        pathname=thispath(1:pathsp(end));
        cd(pathname)
        structArray = feval(fname,structArray);
    end
    if length(w)>0
        cd(origPath)
    end
    varargout{1} = structArray;
    
%------------------------------------------------------------------------
% p = sptool('getprefs',panelName)
% p = sptool('getprefs',panelName,fig)
%   Return preference structure for panel with panelName
%   Inputs:
%     panelName - string
%     fig (optional) - figure of SPTool; uses findobj if not given
%   Outputs:
%     p - value structure for this panel
%     p_defaults - default values structure (from sigprefs.mat) for this panel
case 'getprefs'
    % set showhiddenhandles since this might be called from the command line:
    shh = get(0,'showhiddenhandles');
    set(0,'showhiddenhandles','on');
    panelName = varargin{2};
    if nargin > 2
        fig = varargin{3};
    else
        fig = findobj(0,'Tag','sptool');
    end
    ud = get(fig,'userdata');
    allPanels = {ud.prefs.panelName};
    i = findcstr(allPanels,panelName);
    if isempty(i)
        error(sprintf('Sorry, no panel with name ''%s''; available: %s',...
                     panelName, sprintf('\n   ''%s'' ',allPanels{:})))
    end
    %p = prefstruct(ud.prefs(i));
    p = cell2struct(ud.prefs(i).currentValue,ud.prefs(i).controls(:,1));
    varargout{1} = p;
    if nargout > 1
        % Provide the default value to compare against a possible new preference
        fileName='sigprefs.mat';
        if ~isempty(which(fileName)),
            % Load MAT-file containing saved preferences to compare 
            % against current preferences.
            load(fileName);        
            
            % Return a structure with containing the preference names and values.
            varargout{2} = eval(['SIGPREFS.',ud.prefs(i).panelName]);
        else
            % Compare against default (factory) settings stored in the figure's userdata.
            varargout{2} = cell2struct(ud.prefs(i).controls(:,7),... 
                ud.prefs(i).controls(:,1));
        end
        
    end
    set(0,'showhiddenhandles',shh);

        
%------------------------------------------------------------------------
% sptool('setprefs',panelName,p)
% sptool('setprefs',panelName,p,fig)
%   Set preference structure for panel with panelName
%   Inputs:
%     panelName - string
%     p - value structure for this panel
%     fig (optional) - figure of SPTool; uses findobj if not given
case 'setprefs'
    panelName = varargin{2};
    p = varargin{3};
    if nargin > 3
        fig = varargin{4};
    else
        fig = findobj(0,'Tag','sptool');
    end
    ud = get(fig,'userdata');
    allPanels = {ud.prefs.panelName};
    i = findcstr(allPanels,panelName);
    if isempty(i)
        error(sprintf('Sorry, no panel with name ''%s''; available: %s',...
                     panelName, sprintf('\n   ''%s'' ',allPanels{:})))
    end
    ud.prefs(i).currentValue = struct2cell(p);
    setsigpref(ud.prefs(i).panelName,p);
    set(fig,'userdata',ud)
        
case 'resize'
    sptoolfig = findobj(0,'Tag','sptool');
    ud = get(sptoolfig,'UserData');
    numComponents = length(ud.components);
    fp = get(sptoolfig,'position');
    upperlefty = fp(2)+fp(4);
    
    % NOTE: Only set the figure position when the figure gets too small.
    % Changing the figure's position using SET clears the 'Maximized' state
    % of the figure, hence, disabling the 'Minimize' button on the figure.
    if fp(3) < 208 & fp(4) < 193,
       fp(2) = upperlefty - 193;
       fp(3) = 208;
       fp(4) = 193;
       set(sptoolfig,'Position',fp);
       % warndlg('Restoring SPTool figure to its minimum height and width.','Too Small');
    elseif fp(3) < 208,
       fp(3) = 208;
        set(sptoolfig,'Position',fp);
       % warndlg('Restoring SPTool figure to its minimum width.','Too Thin');
    elseif fp(4) < 193.
       fp(2) = upperlefty - 193;
       fp(4) = 193;
       set(sptoolfig,'Position',fp);
       % warndlg('Restoring SPTool figure to its minimum height.','Too Short');
    end
   
    d1 = 3;
    d2 = 3;
    d3 = d2;
    uw = (fp(3)-(numComponents+1)*d1)/numComponents;
    uh = 20;  % height of buttons and labels
    lb = d2+(d2+uh)*(ud.maxVerbs);  % list bottom
    for i=1:numComponents
        set(ud.list(i),'position',[d1+(d1+uw)*(i-1) lb uw fp(4)-lb-2*d2-uh])
        set(ud.label(i),'position',[d1+(d1+uw)*(i-1) fp(4)-d2-uh uw uh])
        for j=1:length(ud.components(i).verbs)
            set(ud.buttonHandles(i).h(j),'position',...
                    [d1+(d1+uw)*(i-1) lb-j*(d2+uh) uw uh]);
        end
    end

case 'open'
%  sptool('open')   <-- uses uigetfile to get filename and path
%  sptool('open',f) <-- uses file with filename f on the MATLAB path
%  sptool('open',f,p) <-- uses file with filename f and path p
    sptoolfig = findobj(0,'Tag','sptool');
    ud = get(sptoolfig,'UserData');
    if ud.unchangedFlag == 0
        if ~saveChangesPrompt(ud.sessionName,'opening')
            return
        end
    end

    switch nargin 
    case 1
        matlab_wd = pwd;
        cd(ud.wd)
        [f,ud.sessionPath]=uigetfile('*.spt','Open Session');
        cd(matlab_wd)
        loadName = fullfile(ud.sessionPath,f);
        if ~isequal(f,0)
            ud.wd = ud.sessionPath;
            set(gcf,'userdata',ud)
        end
    case 2
        f = varargin{2};
        loadName = which(f);
        if isempty(loadName)
            error(['File ' f ' not found.'])
        end
        ud.sessionPath = '';
    case 3
        f = varargin{2};
        ud.sessionPath = varargin{3};
        loadName = fullfile(ud.sessionPath,f);
        if exist(loadName)~=2
            error(['File ' loadName ' not found.'])
        end
    end

    if ~isequal(f,0)
        load(loadName,'-mat')
        if ~exist('session','var')  % variable
            waitfor(msgbox('Sorry, this file is not a valid SPT file.',...
               'Missing Session Info','error','modal'))
            return
        end
        ud = get(gcf,'userdata');
        ud.sessionName = f;
        ud.session = struct2cell(session);
        [ud.session,msgstr] = sptvalid(ud.session,ud.components);                
        figname = prepender(['SPTool: ' ud.sessionPath ud.sessionName]);
        set(gcf,'name',figname)
        set(gcf,'userdata',ud)
        for k = 1:length(ud.session)
            if ~isempty(ud.session{k})
                set(ud.list(k),'Value',1)
            end
        end
        updateLists
        selectionChanged(sptoolfig,'new')
        for idx = [4 6]
           set(ud.filemenu_handles(idx),'Enable','on');
        end
        set(ud.filemenu_handles(5),'Enable','off');
        ud = get(gcf,'userdata');
        if isempty(msgstr)
            ud.savedFlag = 1;
            ud.unchangedFlag = 1;
        else
            ud.savedFlag = 0;
            ud.unchangedFlag = 0;
        end
        ud.changedStruc = [];
    end
    set(sptoolfig,'UserData',ud);
    
%----------------------------------------------------------------------
%err = sptool('save')
%    save session, using known file name
%    If the session has never been saved, calls sptool('saveas') 
%  CAUTION: saves userdata on exit (to save ud.unchangedFlag)
%  Outputs:
%    err - ==1 if cancelled, 0 if save was successful.
case 'save'
    sptoolfig = findobj(0,'Tag','sptool');
    ud = get(sptoolfig,'UserData');
    if ~ud.savedFlag
        err = sptool('saveas');
        ud = get(sptoolfig,'UserData');
    else
        session = cell2struct(ud.session,{ud.components.name});
        save(fullfile(ud.sessionPath,ud.sessionName),'session')
        err = 0;
    end
    ud.unchangedFlag = ~err;
    if ud.unchangedFlag
        set(ud.filemenu_handles(5),'Enable','off')
    end
    set(sptoolfig,'UserData',ud)
    varargout{1} = err;
%----------------------------------------------------------------------
%err = sptool('saveas')
%    save session, prompting for file name.
%  CAUTION: saves userdata on exit (to save ud.unchangedFlag)
%  Outputs:
%    err - ==1 if cancelled, 0 if save was successful.
case 'saveas'
    sptoolfig = findobj(0,'Tag','sptool');
    ud = get(sptoolfig,'UserData');
    matlab_wd = pwd;
    cd(ud.wd)
    [f,p] = sptuiputfile(ud.sessionName,'Save Session');
	
    cd(matlab_wd)
    if ~isequal(f,0)
        if (length(f)<4) | ~isequal(f(end-3:end),'.spt')
            errstr = {'Sorry, the filename must end in ".spt".'};
            waitfor(msgbox(errstr,'Bad Extension','error','modal'))
        else
            session = cell2struct(ud.session,{ud.components.name});
            ud.sessionName = f;
            ud.sessionPath = p;
            save(fullfile(p,f),'session')
            
            ud.sessionName = f;
            set(sptoolfig,'userdata',ud)
        
            figname = prepender(['SPTool: ' ud.sessionName]);
            set(sptoolfig,'name',figname)
            ud.unchangedFlag = 1;
        end
        ud.wd = p;
    else
        % ud.unchangedFlag = ud.unchangedFlag;  % value doesn't change
    end
    if ud.unchangedFlag
        set(ud.filemenu_handles(5),'Enable','off')
    end
    ud.savedFlag = ud.savedFlag | ud.unchangedFlag;
    set(sptoolfig,'UserData',ud)
    varargout{1} = ~ud.unchangedFlag;
    
case 'pref'
    fig = gcf;
    ud = get(fig,'userdata');
    [ud.prefs ud.panelInd] = sptprefs(ud.prefs,ud.panelInd);
    set(fig,'userdata',ud)

case 'close'
    sptoolfig = findobj(0,'Tag','sptool');
    ud = get(sptoolfig,'UserData');     
    
    for i = 1:length(ud.prefs)
        [p{i},p_default{i}] = sptool('getprefs',...
		                      ud.prefs(i).panelName,sptoolfig);
    end
    if ~isequal(p,p_default) 
        setsigpref({ud.prefs.panelName},p,1);
    end    
 
    if ud.unchangedFlag == 0
        if ~saveChangesPrompt(ud.sessionName,'closing')
            return
        end
    end
    
    for i=1:length(ud.components)
        for j=1:length(ud.components(i).verbs)
            feval(ud.components(i).verbs(j).owningClient,'SPTclose',...
                  ud.components(i).verbs(j).action);
        end
    end
    if ~isempty(ud.importSettings)
        if ishandle(ud.importSettings.fig)
            delete(ud.importSettings.fig) 
        end
    end
    delete(sptoolfig)

case 'create'
% Create Signal, Filter or Spectrum structure from the command line:
% sptool('create',...)
%

    error(nargchk(2,9,nargin))
    shh = get(0,'ShowHiddenHandles');
    set(0,'ShowHiddenHandles','on')
    sptoolfig = findobj(0,'Tag','sptool');
    set(0,'ShowHiddenHandles',shh);
    
    if isempty(sptoolfig)  % SPTool is not open - only get required info
        ud.components = [];
        ud.components = sptcompp(ud.components); % calls one in signal/private
        ud.prefs = sptprefp;
        allPanels = {ud.prefs.panelName};
        plugInd = findcstr(allPanels,'plugins');
        plugPrefs = cell2struct(ud.prefs(plugInd).currentValue,...
                                ud.prefs(plugInd).controls(:,1));
        if plugPrefs.plugFlag
            % now call each one found on path:
            ud.components = sptool('callall','sptcomp',ud.components);
        end
    else                   % SPTool is open; get the info from SPTool
        ud = get(sptoolfig,'UserData');
    end
    compNames = {ud.components.structName};
    
    if ~isstr(varargin{2}) & ~isnumeric(varargin{2})
        set(0,'showhiddenhandles',shh)
        errstr = sprintf(['Second argument must be a string indicating'...
                ' the component name, \nor a double containing data for'...
                ' the first component.']);
        error(errstr)
    end
        
    compIndx = [];
    if isnumeric(varargin{2})
        % No component name specified; use default component (1st one)
        compIndx = 1;
        varargin{end+1} = varargin{end}; % Make room to insert comp name
        mvIndx = 2:nargin; 
        varargin(mvIndx+1) = varargin(mvIndx);
        varargin{2} = compNames{compIndx}; % Insert component name
    else
        % Since the 2nd arg is not data for the default (1st) component
        % then it must be a string containing the component name; check it!
        for i = 1:length(compNames)
            if strcmp(lower(varargin{2}),lower(compNames{i}))
                % Index into compNames which matches compnt name entered
                compIndx = i; 
                break
            end
        end
        if isempty(compIndx)
            % Component name was specified incorrectly
            set(0,'showhiddenhandles',shh)
            str = ['Sorry, no component with name ''%s''; '...
                    'must be one of the following:\n%s'];
            errstr = sprintf(str, varargin{2},...
                     sprintf('      ''%s''\n',compNames{:}));
            error(errstr);
        end
    end

    [popupString,fields,FsFlag,defaultLabel] = ...
        feval(ud.components(compIndx).importFcn,'fields');
    
    [formIndx, formTags] = formIndex(fields,varargin{3});
    if isempty(formIndx)
        set(0,'showhiddenhandles',shh)
        error(['Sorry, the form must be one of the following: ',...
                formTags])
    elseif length(fields) ~= 1  % A valid 'form' was specified
        varargin(3) = [];       % Remove 'form' from input argument list
    end 
    
    if isstr(varargin{end})
        compCell = varargin(3:end-1); % Cell array containing the component
    else
        compCell = varargin(3:end);
    end
    numCompFields = length(fields(formIndx).fields);
    
    if length(compCell) < numCompFields
        % Padd with []s in case some default arguments were left out
        compCell{numCompFields} = [];
    elseif length(compCell) > numCompFields+1
        errstr = sprintf(['Too many input arguments to import a ',...
                '''%s''. '], compNames{compIndx});
        if ~isstr(varargin{end})
            errstr = [errstr, 'Last argument must be a string.'];
        end
        set(0,'showhiddenhandles',shh)            
        error(errstr)
    end
    
    % If Fs required and no Fs was entered; use default Fs=1
    if FsFlag & (length(compCell) < numCompFields+1)  
        Fs = 1;
        compCell = {compCell{:} Fs};
    end
    
    % Complete cell array to be passed to the import function
    paramsCell = {formIndx compCell{:}};  
    [err,errstr,struc] = feval(ud.components(compIndx).importFcn,...
        'make',paramsCell);
    if err
        set(0,'showhiddenhandles',shh)        
        error(errstr)
    end
    
    if isstr(varargin{end})
        if ~isvalidvar(varargin{end})
            set(0,'showhiddenhandles',shh)
            error('Sorry, the label must be a valid MATLAB variable name.')
        else
            % Component label
            struc.label = varargin{end};
        end
    else
        struc.label = defaultLabel;
    end
    
    if nargout == 1
        varargout{1} = struc;
    end
    
case 'load'
% Importing Signals, Filters or Spectra from the command line:
% sptool('load',struc) checks for validity of struc and imports it
%   if it is valid.
% The following syntax creates new structure using 'make' facility of
% importFcn, and imports it into SPTool.

    error(nargchk(2,9,nargin))
    shh = get(0,'showhiddenhandles');
    set(0,'showhiddenhandles','on')
    sptoolfig = findobj(0,'Tag','sptool');
    if isempty(sptoolfig)           % SPTool is closed - open it!
        sptool('init')
        sptoolfig = findobj(0,'Tag','sptool');
    end

    ud = get(sptoolfig,'UserData');
    compNames = {ud.components.structName};
    
    if isstruct(varargin{2})  % Second input is a structure
        struc = varargin{2};
        if nargin > 2
            set(0,'showhiddenhandles',shh)
            error('Too many arguments for importing a structure.')
        end

        errstr = isvalidstruc(struc,ud.components,compNames);
        if ~isempty(errstr)
            set(0,'showhiddenhandles',shh)
            error(errstr)
        end
    else                      % Second input is not a structure
        if ~isstr(varargin{2}) & ~isnumeric(varargin{2})
            set(0,'showhiddenhandles',shh)
            errstr = sprintf(['Second argument must be:\n'...
                    '  a structure, \n'...
                    '  a string indicating the component name, or \n'...
                    '  a double containing data for the first component.'])...

            error(errstr)
        end
        % Create and load at the same time? - 'create' will parse the input
        struc =  sptool('create',varargin{2:end});
    end

    if nargout == 1
        varargout{1} = struc;
    end
    sptool('import',struc,1,sptoolfig)
    set(0,'showhiddenhandles',shh)

case 'import'
% sptool('import')  import structure using dialog box from sptimport
% sptool('import',struc)  import given structure
% sptool('import',struc,selectFlag)  import given structure and change
%   selection in appropriate column to the imported struc if
%     selectFlag == 1 (selectFlag defaults to 0)
% sptool('import',struc,selectFlag,sptoolfig)  
%    uses sptoolfig as figure handle of sptool; if omitted, uses findobj
% sptool('import',struc,selectFlag,sptoolfig,updateChangedStruc)
%   if updateChangedStruc == 1, changedStruc is updated in the case of
%    a replacement of struc (changedStruc is ALWAYS cleared when struc
%    is new).
%   if updateChangedStruc == 0, changedStruc is untouched (default)  

    if nargin < 4
        sptoolfig = findobj(0,'Tag','sptool');
    else
        sptoolfig = varargin{4};
    end    
    ud = get(sptoolfig,'UserData');

    if nargin < 5
        updateChangedStruc = 0;
    else
        updateChangedStruc = varargin{5};
    end

    if nargin < 3
        selectFlag = 0;
    else         
        selectFlag = varargin{3};
    end

    if isempty(sptoolfig)
        error('Sorry, SPTool is not open.')
    end
    labelList = sptool('labelList',sptoolfig);
    if nargin == 1
        [componentNum,struc,ud.importSettings,ud.importwd] = ...
              sptimport(ud.components,labelList,ud.importSettings,ud.importwd);
        figure(sptoolfig)
    
        if componentNum<1  % user cancelled - so save userdata and bail
            set(sptoolfig,'userdata',ud)
            return
        end
    else
        struc = varargin{2};
    end

    componentNum = findcstr({ud.components.structName},...
                            struc.SPTIdentifier.type);

    if ~isempty(findcstr(labelList,struc.label))
    % replace old structure with this label -----------------------------

        % first find column number of old structure:
        for i=1:length(ud.components)
            ind = findStructWithLabel(ud.session{i},struc.label);
            if ~isempty(ind)
                oldComponentNum = i;
                break
            end
        end

        if (nargin==1) | updateChangedStruc
            ud.changedStruc = ud.session{oldComponentNum}(ind);
        end
        if componentNum == oldComponentNum  % replace in same column
            oldComponent = ud.session{oldComponentNum}(ind);
            if (nargin==1) % | strcmp(oldComponent.label,struc.label)

                % The following doesn't work because filtdes.m calls
                % sptool('import') when the filter is already imported
                %if strcmp(oldComponent.label,struc.label)
                %    s1 = sprintf('Warning: Component structure %s ',struc.label);
                %    s2 = sprintf('already exists in SPTool; replacing %s.',struc.label);
                %    disp([s1 s2])
                %end

                % obtained by import dialogue so retain lineinfo field
                ud.session{oldComponentNum}(ind) = ...
                    feval(ud.components(oldComponentNum).importFcn,'merge',...
                      oldComponent,struc);

            else
                ud.session{oldComponentNum}(ind) = struc;
            end
        else  % overwrite object of different type (column)
            ud.session{oldComponentNum}(ind) = [];
            if isempty(ud.session{componentNum})
                ud.session{componentNum} = struc;
            else
                ud.session{componentNum}(end+1) = struc;
            end
        end

        set(sptoolfig,'userdata',ud)
        updateLists(sptoolfig,componentNum)
        if componentNum ~= oldComponentNum
            updateLists(sptoolfig,oldComponentNum)
            set(ud.list(componentNum),'value',length(ud.session{componentNum}))
        end
        if selectFlag | (nargin==1) | updateChangedStruc
            selectionChanged(sptoolfig,'new')
        end
    else
    % append structure to appropriate structure array -------------------
        ud.changedStruc = [];
        if isempty(ud.session{componentNum})
            ud.session{componentNum} = struc;
        else
            ud.session{componentNum}(end+1) = struc;
        end
        set(sptoolfig,'userdata',ud)
        updateLists(sptoolfig)
        set(ud.list(componentNum),'value',length(ud.session{componentNum}))
        selectionChanged(sptoolfig,'new')
    end
    for idx = 4:6
      set(ud.filemenu_handles(idx),'Enable','on');
    end
    ud = get(sptoolfig,'userdata');  % need to get this again since
                                     % selectionChanged call might have 
                                     % changed userdata
    ud.unchangedFlag = 0;
    set(ud.filemenu_handles(5),'Enable','on')
    set(sptoolfig,'UserData',ud);

% sptool('export')
%   Export objects from SPTool.
case 'export'
    sptoolfig = findobj(0,'Tag','sptool');
    ud = get(sptoolfig,'UserData');
    [newprefs,componentSelect,fname,pathname] = ...
        sptexport(ud.prefs,ud.components,ud.session,get(ud.list,'value'),ud.exportwd);
    
    if ~isequal(ud.prefs,newprefs)
        ud.prefs = newprefs;
    end
    
    if ~isequal(pathname,ud.exportwd)
        ud.exportwd = pathname;
        set(sptoolfig,'UserData',ud)
    end

%------------------------------------------------------------------------
% labelList = sptool('labelList',SPTfig)
%   returns a cell array of strings containing all of the
%   object labels currently in the SPTool.
case 'labelList'
    fig = varargin{2};
    labelList = {};
    ud = get(fig,'userdata');
    for i=1:length(ud.components)
        if ~isempty(ud.session{i})
            labelList = {labelList{:} ud.session{i}.label};
        end
    end
    varargout{1} = labelList;

%------------------------------------------------------------------------
% sptool('edit')
% callback of Edit menu
case 'edit'
    sptoolfig = findobj(0,'Tag','sptool');
    ud = get(sptoolfig,'UserData');
    mh2 = ud.editmenu_handles(2);
    mh3 = ud.editmenu_handles(3);
    mh4 = ud.editmenu_handles(4);
    mh5 = ud.editmenu_handles(5);
    str = cell(0);
    for idx = 1:length(ud.components)
           tmp = get(ud.list(idx),'Value');
           if tmp == 1 & isempty(get(ud.list(idx),'String'))
              tmp = [];
           end
           if idx == 1,
              selobj = tmp(:);
              compidx = [ones(length(tmp),1)];
           else
              selobj = [selobj; tmp(:)];
              compidx = [compidx; idx*ones(length(tmp),1)];
           end
           
           ud.compidx = [compidx selobj];
           tmpstr = get(ud.list(idx),'String');
           if ~isempty(tmpstr)
              tmpstr = tmpstr(tmp);
              if idx == 1
                  str = tmpstr;
              else
                  str = cat(1,str,tmpstr);
              end
           end
    end
    
    if ~isempty(selobj)
         % first initialize FsFlag array (determines if you can edit the
         % sampling frequency of a component or not)
         for i = 1:length(ud.components)
            [popupString,fields,FsFlag(i),defaultLabel] = ...
                 feval(ud.components(i).importFcn,'fields');
         end
         
         set(mh2,'Enable','on');
         set(mh3,'Enable','on');
         set(mh4,'Enable','on');
         set(mh5,'Enable','on');
         
         % remove extra menu items:
         delete(ud.dupsub(length(selobj)+1:end))
         ud.dupsub(length(selobj)+1:end)=[];
         delete(ud.clearsub(length(selobj)+1:end))
         ud.clearsub(length(selobj)+1:end)=[];
         delete(ud.namesub(length(selobj)+1:end))
         ud.namesub(length(selobj)+1:end)=[];
         ind = length(selobj)+1:length(ud.freqsub);
         delete(ud.freqsub(ind( find(ishandle(ud.freqsub(ind))) )))
         ud.freqsub(length(selobj)+1:end)=[];
         
         for idx1 = 1:length(selobj)
             if idx1 > length(ud.dupsub)
             % create a new uimenu
                ud.dupsub(idx1) = uimenu(mh2,'Label',str{idx1},'tag','dupmenu',...
                     'Callback',['sptool(''duplicate'',',int2str(idx1),')']);
                ud.clearsub(idx1) = uimenu(mh3,'Label',str{idx1},'tag','clearmenu',...
                     'Callback',['sptool(''clear'',',int2str(idx1),')']);
                ud.namesub(idx1) = uimenu(mh4,'Label',str{idx1},...
                     'tag','newnamemenu',...
                     'Callback',['sptool(''newname'',',int2str(idx1),')']);
                if FsFlag(compidx(idx1))
                    ud.freqsub(idx1) = uimenu(mh5,'Label',str{idx1},...
                        'tag','freqmenu','Callback',['sptool(''freq'',',int2str(idx1),')']);
                else
                % just put place holder here - don't create menu item since
                % we can't edit the Sampling frequency for this component
                    ud.freqsub(idx1) = -1;
                end
             end
             if idx1 <= length(selobj) 
             % change label and ensure visibility of existing uimenu
                set(ud.dupsub(idx1),'Visible','on','Label',str{idx1});
                set(ud.clearsub(idx1),'Visible','on','Label',str{idx1});
                set(ud.namesub(idx1),'Visible','on','Label',str{idx1});
                if FsFlag(compidx(idx1))
                    if ishandle(ud.freqsub(idx1))
                        set(ud.freqsub(idx1),'Visible','on','Label',str{idx1});
                    else
                        ud.freqsub(idx1) = uimenu(mh5,'Label',str{idx1},...
                             'tag','freqmenu', ...
                             'Callback',['sptool(''freq'',',int2str(idx1),')']);
                    end
                elseif ishandle(ud.freqsub(idx1))
                        set(ud.freqsub(idx1),'Visible','off')
                end
             end
 
         end
    else
         set(mh2,'Enable','off');
         set(mh3,'Enable','off');
         set(mh4,'Enable','off');
         set(mh5,'Enable','off');
    end
    drawnow;
    set(sptoolfig,'Userdata',ud);
    
%------------------------------------------------------------------------
% sptool('duplicate')
% callback of duplicate submenu
case 'duplicate'
    sptoolfig = findobj(0,'Tag','sptool');
    ud = get(sptoolfig,'UserData');
    lab = get(ud.dupsub(varargin{2}),'Label');
    bracket = findstr(lab,'[');
    lab1 = [lab(1:bracket-2),'copy'];
    lab2 = lab(bracket-1:end); 
    idx = ud.compidx(varargin{2},:);
    
    % make sure new label is unique:
    labelList = {ud.session{idx(1)}.label};
    numCopies = 1;
    while ~isempty(findcstr(labelList,lab1))
        lab1 = [lab(1:bracket-2),'copy',num2str(numCopies)];
        numCopies = numCopies + 1;
    end
  
    ud.changedStruc = [];  
    ud.session{idx(1)}(end+1)=ud.session{idx(1)}(idx(2));
    ud.session{idx(1)}(end).label = lab1;
    n = length(get(ud.list(idx(1)),'String'));
    set(ud.list(idx(1)),'Value',n+1);
    ud.unchangedFlag = 0;
    set(ud.filemenu_handles(5),'Enable','on')
    set(sptoolfig,'UserData',ud);
    updateLists(sptoolfig)
    selectionChanged(sptoolfig,'dup')
    
%------------------------------------------------------------------------
% sptool('clear')
% callback of clear submenu
case 'clear'
    sptoolfig = findobj(0,'Tag','sptool');
    ud = get(sptoolfig,'UserData');
    lab = get(ud.dupsub(varargin{2}),'Label');
    idx = ud.compidx(varargin{2},:);
    
    ud.changedStruc = ud.session{idx(1)}(idx(2));
    ud.session{idx(1)}(idx(2)) = [];
    
    if ud.components(idx(1)).multipleSelection
    % just remove item from selection
        listVal = get(ud.list(idx(1)),'value');
        listValInd = find(listVal==idx(2));
        listVal(listValInd+1:end) = listVal(listValInd+1:end)-1;
        listVal(listValInd) = [];
    else
        listVal = 1;
    end
    str = get(ud.list(idx(1)),'String');
    str(idx(2)) = [];
    set(ud.list(idx(1)),'Value',listVal,'String',str);    
    
    ud.unchangedFlag = 0;
    set(ud.filemenu_handles(5),'Enable','on')
    set(sptoolfig,'UserData',ud)
    selectionChanged(sptoolfig,'clear')
    
%------------------------------------------------------------------------
% sptool('newname')
% callback of Name... submenu
case 'newname'
    sptoolfig = findobj(0,'Tag','sptool');
    ud = get(sptoolfig,'UserData');
    lab = get(ud.dupsub(varargin{2}),'Label');
    bracket = findstr(lab,'[');
    lab1 = lab(1:bracket-2);
    lab2 = lab(bracket-1:end);
    idx = ud.compidx(varargin{2},:);
    prompt={'Enter new variable name:'};
    def = {lab1};
    title = 'Name Change';
    lineNo = 1;
    lab1 =inputdlg(prompt,title,lineNo,def);
    if isempty(lab1)
        return
    end            
    err = ~isvalidvar(lab1{:});
    if ~err
        currentlabels = get(ud.list(idx(1)),'String');
        labelList = sptool('labelList',sptoolfig);
        if ~isempty(findcstr(labelList,deblank(lab1{:})))
               % error prompt
               errstr = {'There is already an object in the SPTool '
                         'with this name.  Please use a unique name.'};
               errordlg(errstr,'Non-unique Name','replace');
               return
         end   
    else
         errstr = {'Sorry, the name you entered is not valid.'
                   'Please use a legal MATLAB variable name.'};
         errordlg(errstr,'Bad Variable Name','replace');
         return
    end     

    if isequal(ud.session{idx(1)}(idx(2)).label,lab1{:})
        % new label is the same as the old one - do nothing!
        return
    end
    ud.changedStruc = ud.session{idx(1)}(idx(2));
    ud.session{idx(1)}(idx(2)).label = lab1{:};
    ud.unchangedFlag = 0;
    set(ud.filemenu_handles(5),'Enable','on')
    set(sptoolfig,'UserData',ud);
    listStr = get(ud.list(idx(1)),'string');
    listStr{idx(2)} = [deblank(lab1{:}) lab2];
    set(ud.list(idx(1)),'string',listStr)
    selectionChanged(sptoolfig,'label')
    
%------------------------------------------------------------------------
% sptool('freq')
% callback of Sampling Frequency... submenu
case 'freq'
    sptoolfig = findobj(0,'Tag','sptool');
    ud = get(sptoolfig,'UserData');
    lab = get(ud.dupsub(varargin{2}),'Label');
    idx = ud.compidx(varargin{2},:);
    prompt={'Enter the sampling frequency.'};
    def = {sprintf('%.9g',ud.session{idx(1)}(idx(2)).Fs)};
    title = 'Sampling Frequency for Signal Processing';
    lineNo = 1;
    Fs=inputdlg(prompt,title,lineNo,def);
    if isempty(Fs)
        return
    end
    [Fs,err] = validarg(Fs{:},[0 Inf],[1 1],'sampling frequency (or expression)');
    if err ~= 0
         return
    end
    if ud.session{idx(1)}(idx(2)).Fs == Fs
        % new Fs is the same as the old one - do nothing!
        return
    end
    
    ud.changedStruc = ud.session{idx(1)}(idx(2));
    ud.session{idx(1)}(idx(2)) = feval(ud.components(idx(1)).importFcn,'changeFs',...
                  ud.session{idx(1)}(idx(2)),Fs);
    ud.unchangedFlag = 0;
    set(ud.filemenu_handles(5),'Enable','on')
    set(sptoolfig,'UserData',ud);
    selectionChanged(sptoolfig,'Fs')

%------------------------------------------------------------------------
% sptool('help','overview')  <-- overview help
% sptool('help') <-- context sensitive help
case 'help'
    sptoolfig = findobj(0,'Tag','sptool');
    ud = get(sptoolfig,'UserData');
    whichHelp = varargin{2};
    titleStr = 'SPTOOL Help';
    helpFcn = 'spthelpstr';
    if strcmp(whichHelp,'overview')
       spthelp('tag',sptoolfig,titleStr,helpFcn,'overview');
       return
    end
    saveEnableControls = [];
    if ud.pointer ~= 2   % if not in help mode
        numComponents = length(ud.components);
        % enter help mode
        controlNumber = 0;
        for idx1 = 1:numComponents
            controlNumber = controlNumber + 1;
            saveEnableControls(controlNumber) = ud.list(idx1);
            for idx2 = 1:length(ud.components(idx1).verbs)
                controlNumber = controlNumber + 1;
                saveEnableControls(controlNumber) = ud.buttonHandles(idx1).h(idx2);
            end
        end
        spthelp('enter',sptoolfig,saveEnableControls,[],titleStr,helpFcn)
    else
        spthelp('exit')
    end

%------------------------------------------------------------------------
%  sptool('verb',i,j)  - i component #, j verb #
case 'verb'
    sptoolfig = findobj(0,'Tag','sptool');
    setptr(sptoolfig,'watch')
    drawnow
    ud = get(sptoolfig,'UserData');
    i = varargin{2};
    j = varargin{3};
    feval(ud.components(i).verbs(j).owningClient,'action',...
          ud.components(i).verbs(j).action);
    setptr(sptoolfig,'arrow')
    
%------------------------------------------------------------------------
%  sptool('list',i)  - i component #     LISTBOX CALLBACK
case 'list'
    sptoolfig = findobj(0,'Tag','sptool');
    ud = get(sptoolfig,'UserData');
    idx = varargin{2};
    if 0  % short cut (double click) is disabled for now
    % if strcmp(get(sptoolfig,'SelectionType'),'open')
       dc = ud.components(idx).defaultClient;
       whichClient = 0;
       k = 0;
       while ~whichClient
           k = k + 1;
           if strcmp(dc,ud.components(idx).verbs(k).owningClient)
              whichClient = k;
           end
       end
       sptool('verb',idx,whichClient);
    else
       selectionChanged(sptoolfig,'value')
    end
    
%----------------------------------------------------------------------------
% struct = sptool('changedStruc',sptoolfig) - return recently
%  changed structure (removed, imported, name or Fs changed)
%  sptoolfig - optional, found with findobj if not present
case 'changedStruc'
    if nargin < 2
        shh = get(0,'ShowHiddenHandles');
        set(0,'ShowHiddenHandles','on')
        sptoolfig = findobj(0,'Tag','sptool');
        set(0,'ShowHiddenHandles',shh);
    else
        sptoolfig = varargin{2};
    end
    ud = get(sptoolfig,'UserData');
    varargout{1} = ud.changedStruc;
    
%----------------------------------------------------------------------------
% Fs  = sptool('commonFs)
case 'commonFs'         
   % For sampling frequency, if there are filters in the SPTool,
        % use their common sampling frequency (or the last one if
        % they don't have one).  If there no filters, do the same
        % for Signals.  If no signals either, use Fs = 1.
        ftemp = sptool('Filters');
        if ~isempty(ftemp)
            Fs = ftemp(end).Fs;
        else
            stemp = sptool('Signals');
            if ~isempty(stemp)
                Fs = stemp(end).Fs;
            else
                Fs = 1;
            end
        end
        varargout{1} = Fs;
  
%----------------------------------------------------------------------------
%  Client API
%  can ask for the currently selected objects
% [s,ind] = sptool(componentName,allFlag)
% [s,ind] = sptool(componentName,allFlag,fig)
% Returns a structure array of the current data items in the SPTool
%   Inputs:
%      componentName - string; name of component e.g. 'Filters' or 'Signals'
%      allFlag - 1 ==> return all objects of the requested type in s
%                0 ==> return only the currently selected objects in s
%                allFlag is optional; it defaults to 1
%      fig - figure handle of SPTool; optional - if omitted, will be found with findobj
%   Outputs:
%       s - structure array
%       ind - optional output; indices of s which are currently selected 
%             in SPTool
otherwise
    sptoolfig = [];
    if nargin < 3
        shh = get(0,'ShowHiddenHandles');
        set(0,'ShowHiddenHandles','on')
        sptoolfig = findobj(0,'Tag','sptool');
        set(0,'ShowHiddenHandles',shh);
    else
        if isempty(varargin{3}) | ishandle(varargin{3})
            sptoolfig = varargin{3};
        else
            error('Invalid input.  Please type ''help sptool'' for more information.')
        end
    end
    
    if isempty(sptoolfig)
        error('Sorry, SPTool is not open.')
    end
    
    ud = get(sptoolfig,'UserData');
    l = {ud.components.name};
    whichComponent = findcstr(l,action);
    if isempty(whichComponent)
        error(sprintf('Sorry, no component with name ''%s''; available: %s',...
                     action, sprintf('''%s'' ',l{:})))
    end
    if nargin > 1
        allFlag = varargin{2};
    else
        allFlag = 1;
    end
    
    varargout{1} = ud.session{whichComponent};
    ind = get(ud.list(whichComponent),'value');
    if allFlag
        varargout{2} = ind;
    else
        if isempty(varargout{1})
            varargout = { [] [] };
        else
            varargout = {varargout{1}(ind) 1:length(ind)};
        end
    end
    
end


%-------------------------------------------------------------------------
function ud = init_sptool_data_settings
% INIT_SPTOOL_DATA_SETTINGS Build the preference structure to create 
%                           uicontrols for the preference dialog window.  
%                           Also create the "factory default" settings in 
%                           the .controls field.

    ud.prefs = sptprefp;
    
    % Loop through each panel of preferences
    for i=1:length(ud.prefs)
        p = getsigpref(ud.prefs(i).panelName);
        if ~isempty(p)
            ud.prefs(i).currentValue = struct2cell(p);
            % check to make sure currentValue has the correct number of
            % elements.  If it doesn't then the preferences must be out
            % of date, so set the current value to the factory setting.
            if length(ud.prefs(i).currentValue) ~= length(ud.prefs(i).controls(:,1))
                ud.prefs(i).currentValue = ud.prefs(i).controls(:,7);
                warning(...
                    sprintf(['Preferences for "%s" not correct size and are\n'...
                        'assumed to be out-of-date;' ... 
                        ' using factory settings.'],ud.prefs(i).panelDescription))
            end
        end
    end
    allPanels = {ud.prefs.panelName};
    plugInd = findcstr(allPanels,'plugins');
    plugPrefs = cell2struct(ud.prefs(plugInd).currentValue,ud.prefs(plugInd).controls(:,1));
    if plugPrefs.plugFlag   
        % add any additional preferences
        ud.prefs = sptool('callall','sptpref',ud.prefs);
    end
    
    ud.panelInd = 1;
    ud.components = [];
    ud.components = sptcompp(ud.components); % calls one in signal/private
    if plugPrefs.plugFlag
        % now call each one found on path:
        ud.components = sptool('callall','sptcomp',ud.components);
    end

    ud.sessionName = 'untitled.spt';
    ud.wd = pwd;                 % working directory for Opening, Saving, etc.
    ud.importwd = pwd;           % working directory for importing file data
    ud.exportwd = pwd;           % working directory for exporting data to disk
    ud.savedFlag = 0;            % flag indicating if session has ever been saved
    ud.unchangedFlag = 1;        % indicates if sess. is unchanged since last save
    
    % the 'changeStruc' field will contain the structure which has been
    % altered by the most recent operation, as it was before the operation.
    % operations include: clear, name change, import over (such as 
    % applying a filter), sampling frequency change
    ud.changedStruc = [];
    ud.importSettings = [];
    ud.sessionPath = [];
    ud.pointer = 1;
    
    
%-------------------------------------------------------------------------
function [fig,ud] = create_sptool_gui(ud)
% CREATE_SPTOOL_GUI Creates the GUI for SPTool - the data manager

    numComponents = length(ud.components);
    maxVerbs = 1;
    for i=1:numComponents
        maxVerbs = max(maxVerbs,length(ud.components(i).verbs));
    end
    ud.maxVerbs = maxVerbs;
    
    % Determine figure position
    figHeight   = ud.maxVerbs*20+240;
    screenRect  = get(0,'screensize');
    fp = [18 screenRect(4)-figHeight-50 140*numComponents figHeight];
          
    figname = prepender(['SPTool: ' ud.sessionName]);
    
    uibgcolor = get(0,'defaultuicontrolbackgroundcolor');
    uifgcolor = get(0,'defaultuicontrolforegroundcolor');
 
    % CREATE FIGURE
    fig = figure('createfcn','',...
            'closerequestfcn','sptool(''close'')',...
            'tag','sptool',...
            'numbertitle','off',...
            'integerhandle','off',...
            'units','pixels',...
            'position',fp,...
            'menubar','none',...
            'color',uibgcolor,...
            'inverthardcopy','off',...
            'paperpositionmode','auto',...
            'visible','off',...
            'name',figname);


    fontsize = get(0,'defaultuicontrolfontsize');
    for i=1:numComponents
        maxVerbs = max(maxVerbs,length(ud.components(i).verbs));
        ud.list(i) = uicontrol('style','listbox','backgroundcolor','w',...
                      'units','pixels',...
                      'callback',['sptool(''list'',' num2str(i) ')'],...
                      'value',[],'Tag',['list' num2str(i)]);
        if ud.components(i).multipleSelection
            set(ud.list(i),'max',2)
        end
        ud.label(i) = uicontrol('style','text','string',ud.components(i).name,...
                      'Tag',['list' num2str(i)],...
                      'units','pixels',...
                      'fontweight','bold');
        for j=1:length(ud.components(i).verbs)
            ud.buttonHandles(i).h(j) = uicontrol('string',...
                 ud.components(i).verbs(j).buttonLabel,...
                 'units','pixels',...
                 'callback',['sptool(''verb'',' num2str(i) ',' num2str(j) ')'],...
                 'tag',[ud.components(i).verbs(j).owningClient ':' ...
                        ud.components(i).verbs(j).action]);
        end
    end

    % ====================================================================
    % MENUs
    %  create cell array with {menu label, callback, tag}

 %  MENU LABEL                     CALLBACK                      TAG
fm={ 
 'File'                              ' '                        'filemenu'
 '>&Open Session...^o'               'sptool(''open'')'         'loadmenu'
 '>------'                           ' '                        ' '
 '>&Import...^i'                     'sptool(''import'')'       'importmenu'
 '>&Export...^e'                     'sptool(''export'')'       'exportmenu'
 '>------'                           ' '                        ' '
 '>&Save Session^s'                  'sptool(''save'');'        'savemenu'
 '>Save Session As...'               'sptool(''saveas'');'       'saveasmenu'
 '>------'                           ' '                        ' '
 '>Preferences...'                   'sptool(''pref'') '        'prefmenu'
 '>&Close^w'                         'sptool(''close'')'        'closemenu'};
 
    ud.filemenu_handles = makemenu(fig, char(fm(:,1)),char(fm(:,2)), char(fm(:,3)));
    for idx = 4:6
      set(ud.filemenu_handles(idx),'Enable','off');
    end
 
 
 %  MENU LABEL                     CALLBACK                      TAG
em={ 
 'Edit'                             'sptool(''edit'')'           'editmenu'
 '>Duplicate'                       'sptool(''edit'')'           'dupmenu'
 '>Clear'                           'sptool(''edit'')'           'clearmenu'
 '>------'                          ' '                          ' '
 '>Name...'                         'sptool(''edit'')'           'newnamemenu'  
 '>Sampling Frequency...'           'sptool(''edit'')'           'freqmenu'};

    ud.editmenu_handles = makemenu(fig, char(em(:,1)),char(em(:,2)), char(em(:,3)));
    ud.dupsub(1) = uimenu(ud.editmenu_handles(2),'Label',' ',...
           'tag','dupmenu','Callback',['sptool(''duplicate'',',int2str(1),')']);
    ud.clearsub(1) = uimenu(ud.editmenu_handles(3),'Label',' ',...
           'tag','clearmenu','Callback',['sptool(''clear'',',int2str(1),')']);
    ud.namesub(1) = uimenu(ud.editmenu_handles(4),'Label',' ',...
           'tag','newnamemenu','Callback',['sptool(''newname'',',int2str(1),')']);
    ud.freqsub(1) = uimenu(ud.editmenu_handles(5),'Label',' ',...
           'tag','freqmenu','Callback',['sptool(''freq'',',int2str(1),')']);
    for idx = 2:5
       set(ud.editmenu_handles(idx),'Enable','off');
    end


hm={'Help'                          ' '                          'helpmenu'
    '>Overview...'                  'sptool(''help'',''overview'')'  'helpoverview'
    '>Context Sensitive...^h'       'sptool(''help'',''mouse'')'  'helpmouse'
    };

    ud.helpmenu_handles = makemenu(fig, char(hm(:,1)),char(hm(:,2)), char(hm(:,3)));
    
    
wm={'Window'                        'winmenu(gcf);'              'winmenu'};


   ud.winmenu_handles = makemenu(fig, char(wm(:,1)),char(wm(:,2)), char(wm(:,3)));
   winmenu(fig);

    ud.session = cell(numComponents,1);
    for i=1:numComponents
        ud.session{i} = [];
    end
    
%-------------------------------------------------------------------------
function updateLists(fig,componentNum)
%updateLists  - creates listbox strings for all components
%   based on ud.session

    if nargin<1
        fig = findobj(0,'Tag','sptool');
    end
    ud = get(fig,'UserData');    
    if nargin<2
        componentNum = 1:length(ud.components);
    end
    
    for i=componentNum
        listStr = cell(1,length(ud.session{i}));
        for j=1:length(ud.session{i})
            listStr{j} = ...
              [ud.session{i}(j).label ' [' ud.session{i}(j).type ']'];
        end
        if length(listStr)==0
            set(ud.list(i),'value',[])
        elseif length(listStr)<max(get(ud.list(i),'value'))
            set(ud.list(i),'value',1)
        end
        set(ud.list(i),'string',listStr)
    end
    
    
%-------------------------------------------------------------------------
function ind = findStructWithLabel(structArray,label);
% ind = findStructWithLabel(structArray,label)
% returns the index of the (unique) structure element in structArray with
% field .label equal to the string argument label.

    if isempty(structArray)
        ind = [];
    else
        l = {structArray.label};
        ind = findcstr(l,label);
    end
    
    
%-------------------------------------------------------------------------
function selectionChanged(fig,msg)
%selectionChanged
%   enables / disables all verb buttons based on listbox values;
%   in the process the client tools have a chance to update themselves
%   based on the selection
%   msg - string; will be passed to the clients as well
%     possibilities: 'new' - new entries to list, or change in current entries - default
%                    'value' - new listbox value, no other change
%                    'Fs' - sampling freq of object in current selection changed
%                    'label' - label of an object in the current selection changed
%                    'dup' - an object has been duplicated
%                    'clear' - an object has been deleted  
    
    if nargin<1
        fig = findobj(0,'Tag','sptool');
    end
    if nargin<2
        msg = 'new';
    end
    ud = get(fig,'userdata');
    for i=1:length(ud.components)
        for j=1:length(ud.components(i).verbs)
            enable = feval(ud.components(i).verbs(j).owningClient,'selection',...
                  ud.components(i).verbs(j).action,msg,fig);
            set(ud.buttonHandles(i).h(j),'enable',enable)
        end
    end
    
    
%-------------------------------------------------------------------------
function continue = saveChangesPrompt(sessionName,operation)
% continue = saveChangesPrompt(sessionName,operation)
%    Informs user via a dialog box that the SPTool session
%    with name sessionName has been changed.  
%    The user then has the choice of
%      1) saving the session
%      2) not saving the session
%      3) cancelling the operation
%    continue will be true (1) if the session was saved successfully
%    or the Don't save button was pressed (option 2 above).
% operation should be a string indicating the operation being
% performed.  It should be either 'closing' or 'opening'.  This
% string will actually appear in the dialog box.
%
%    Note: userdata will be changed if session is saved.

   continue = 1;
   ButtonName = questdlg({'Save changes to SPTool Session' ...
           sprintf('''%s'' before %s?',sessionName,operation)},...
               'Unsaved Session','Save','Don''t Save','Cancel','Save');
   switch ButtonName
   case 'Save'
      if sptool('save')
          continue = 0;
      end
   case 'Cancel'
      continue = 0;
   end


%-------------------------------------------------------------------------
function errstr = isvalidstruc(struc,ImportFcns,compNames)
% ISVALIDSTRUC returns an empty string if all the fields in the given
% structure are valid, otherwise it returns a string containing an error
% message.
% Inputs:
%   struc - component structure
%   ImportFcs - structure containing the import functions
%   compNames - component names
% Outputs:
%   errstr - error message if structure is invalid

    errstr = '';
    if ~isfield(struc,'label') | ~isstr(struc.label) | isempty(struc.label)
        errstr = ['Sorry, the structure needs a ''.label'' field '...
                  'containing a non-empty string.'];
    
    elseif ~isvalidvar(struc.label)
        errstr = ['Sorry, the label must be a valid MATLAB variable '...
                   'name.'];
    
    elseif isfield(struc,'SPTIdentifier') & (length(struc) == 1) & ...
            isfield(struc.SPTIdentifier,'type')
        i = find(strcmp(struc.SPTIdentifier.type,compNames));
        if ~isempty(i)
            [valid,struc] = feval(ImportFcns(i).importFcn,'valid',struc);
            if ~valid
                errstr = ['Sorry, this is not a valid SPTool structure.'];
            end
        else
            str = ['Sorry, no component with name ''%s''; '...
                    'must be one of the following:\n%s'];
            compName = struc.SPTIdentifier.type;
            errstr = sprintf(str, compName,...
                     sprintf('      ''%s''\n',compNames{:}));
        end
    end


%-------------------------------------------------------------------------
function [session_out,msgstr] = sptvalid(session_in,components)
% [session,msgstr] = sptvalid(session,components)
%   "Validates" a session by updating all structs, and removing any
%   that are invalid.
%   This function will ensure that the elements in the input session
%   cell array match the components in the input components array, if
%   necessary by reordering, removing, or adding empty elements to
%   the session struct.
% Inputs:
%   session - cell array, each element is a vector of SPTool structs
%   components - structure array, one element for each element in
%      session, specifies the component of the corresponding element in session
% Outputs:
%   session - updated session cell array
%   msgstr - empty if session has not been changed, otherwise contains a 
%            message saying that the input session file was not up-to-date
%            and will need to be saved.
%            If not empty, this message will be displayed in a dialog box

    invalidList = {};
    updatedList = {};
    
    session_out = session_in;

    for i=length(session_out):-1:1
        % removed bad structure arrays:
        if ~isempty(session_out{i}) & ~isfield(session_out{i},'SPTIdentifier')
            session_out(i) = [];
        elseif ~isempty(session_out{i}) & ...
               ~isfield(session_out{i}(1).SPTIdentifier,'type')
            session_out(i) = [];
        end
    end
        
    sessionTypes = {};
    for i=1:length(session_out)
        if ~isempty(session_out{i})
            sessionTypes{i} = session_out{i}(1).SPTIdentifier.type;
        else
            sessionTypes{i} = '';
        end
    end
    componentTypes = {components.structName};
    
    % find and remove sessions not in components:
    [C,i] = setdiff(char(sessionTypes),char(componentTypes),'rows');
    for ii=sort(-i)
       invalidList = addinvalid(invalidList,session_out{-ii});
    end
    
    session_out(i) = [];
    sessionTypes(i) = [];
    
    % find components not in sessions, and add empty lists for those components
    session_temp = session_out;
    session_out = cell(length(components),1);
    [C,ia,ib] = intersect(char(sessionTypes),char(componentTypes),'rows');
    session_out(ib) = session_temp(ia);    

    % now use importfcn of each component to validate / update each 
    % structure
    for i=1:length(session_out)
        objArray = [];
        for j=1:length(session_out{i})
            [valid,s] = feval(components(i).importFcn,'valid',session_out{i}(j));
            if valid
                if isempty(objArray)
                    objArray = s;
                else
                    objArray(end+1) = s;
                end
                if ~isequal(s,session_out{i}(j))
                    updatedList = addinvalid(updatedList,session_out{i}(j));
                end
            else
                invalidList = addinvalid(invalidList,session_out{i}(j));
            end
            % disp(session_out{i}(j).label)
        end
        session_out{i} = objArray;
    end

    if ~isequal(session_in,session_out)
        if isempty(invalidList)
            invalidList = {'<none>'};
        end
        if isempty(updatedList)
            updatedList = {'<none>'};
        end
        msgstr = sprintf(...
        ['This session file was out-of-date or contained invalid\n' ...
         'SPTool structures.\n' ...
         '\n' ...
         'Updated structures: %s\n'...
         'Invalid structures (not loaded): %s\n' ...
         '\n'...
         'The session will need to be saved.'],...
         sprintf('%s ',updatedList{:}),sprintf('%s ',invalidList{:}));
        
        waitfor(msgbox(msgstr,...
               'Session Changes','error','modal'))
    else
        msgstr = '';
    end
    
    
%-------------------------------------------------------------------------
function invalidList = addinvalid(invalidList,struc)
    if ~isempty(struc)
        invalidList{end+1} = [struc.label '(' struc.SPTIdentifier.type ')'];
    else
        invalidList{end+1} = '';
    end
    
    
%-------------------------------------------------------------------------
function [formIndx, formTags] = formIndex(fields,form)
% FORMINDEX determine what the 'form' is for a given component.  Some
% components have different forms. For example filters can be entered as a
% transfer function (tf), state space (ss), second-order sections (sos), or
% zero-pole gain (zpk).
% Inputs:
%   fields - the component structure fields
%   form - the third input argument to sptool which, depending on the
%          component, could be the 'form' which describes how the component
%          was entered eg. for filters it's: 'tf','ss','sos', or 'zpk'
% Outputs:
%   formIndx - a scalar indicating which form was used when entering the
%              component
%   formTags - a list of strings which are possible forms

    formTags = [];
    if length(fields) == 1       % Component only has one form
        formIndx = 1;
    else                         % Component has multiple forms
        for i=1:length(fields)
            formTag = fields(i).formTag;
            formTags = [formTags '''' formTag '''' ' '];
        end
        
        formIndx = [];
        if ~isempty(form)  % Form was specified
            if isstr(form)
                formIndx = find(strcmp(form,{fields.formTag}));
            elseif length(form) == 1 & find(form==[1:length(fields)])
                % Using numbers to specify form
                formIndx = form;
            end
        end
    end
    
    
%-------------------------------------------------------------------------
function [f,p] = sptuiputfile(sessionName,dlgboxTitle)   
% SPTUIPUTFILE The following is a workaround for uiputfile which works
%              differently on the PC, UNIX and MAC.
% PCWIN: uiputfile adds the extension as specified in 'Save as Type'
%        and checks if the file exists.
% UNIX:  uiputfile DOES NOT add the extension as specified in the
%        'Filter', however it DOES check if the file, with the
%        extension specified in the 'Filter', exists.
% MAC:   uiputfile checks if the file name, as entered, exists
%
% Inputs: 
%   sessionName - string containing the default session name
%   dlgboxTitle - string containing the title of the dialog box
% Ouputs: 
%   f - string containing the file name with extension .spt
%   p - string contaning the path to file

    [f,p] = uiputfile(sessionName,dlgboxTitle);

    if ~isequal(f,0) & ~strcmp(computer,'PCWIN') & ...
            ~isequal(f(end-3:end),'.spt')

        % UNIX and MAC's uiputfile DOES NOT automatically add 
        % extensions to file names
        if isempty(findstr(f,'.'))
            % no '.' extension, so add '.spt' to file name
            f = [f '.spt'];
        end
        
        % On the MAC only... after adding the extension to the file name we
        % must check again if the file exists since the MAC's uiputfile
        % only checks the file name as entered (eg without the extension) 
        ButtonName = 'No';
        while ~isequal(f,0) & strcmp(computer,'MAC2') & strcmp(ButtonName,'No') 
            if exist(f,'file') == 2
                question=[p,f,' This file already exists. Replace existing file?'];
                ButtonName = questdlg(question,...
                                      'Save Session',...
                                      'Yes','No','No');  % Default is 'No'
            else
                ButtonName = 'Yes';
            end
            if strcmp(ButtonName,'No')
                [f,p]=uiputfile(f,dlgboxTitle); 
                if ~isequal(f,0) & isempty(findstr(f,'.'))
                    % no '.' extension, so add '.spt' to file name
                    f = [f '.spt'];
                else 
                    %file name with extension has been entered; exit while-loop
                    ButtonName = 'Yes';
                end
            end
        end   % while-loop
    end   % if not PCWIN and no extension specified

% end of file - sptool.m