view m-toolbox/classes/@pzmodel/pzmodel.m @ 44:409a22968d5e default

Add unit tests
author Daniele Nicolodi <nicolodi@science.unitn.it>
date Tue, 06 Dec 2011 18:42:11 +0100
parents f0afece42f48
children a71a40911c27
line wrap: on
line source

% PZMODEL constructor for pzmodel class.
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%
% DESCRIPTION: PZMODEL constructor for pzmodel class.
%
%
% CONSTRUCTOR:
%
%       pzm = pzmodel()                - creates an empty pzmodel object
%       pzm = pzmodel(g, p, z)         - construct from gain, poles, zeros
%       pzm = pzmodel(g, p, z, d)      - construct from gain, poles, zeros and a delay (in seconds)
%       pzm = pzmodel(g, p, z, 'name') - construct including name
%       pzm = pzmodel(g, p, z,         - construct from gain, poles, zeros, and
%                     iunits, ounits)              io-units
%       pzm = pzmodel('foo.fil')       - construct from LISO .fil file
%       pzm = pzmodel('foo.xml')       - construct by loading the pzmodel from disk
%       pzm = pzmodel('foo.mat')       - construct by loading the pzmodel from disk
%       pzm = pzmodel(pl)              - create a pzmodel object from the
%                                        description given in the parameter list.
%       pzm = pzmodel(rat)             - creates a pzmodel from rational TF
% 
% 
% Poles and zeros can be given as single values, or a 2-element vector for
% [f, Q]. Multiple poles and zeros should be given in a cell-array:
% 
% e.g. p = pzmodel(1, {1, 2, [3 4]}, {5, [6 10]}, 2)
%
% <a href="matlab:utils.helper.displayMethodInfo('pzmodel', 'pzmodel')">Parameters Description</a>
%
% VERSION:     $Id: pzmodel.m,v 1.130 2011/08/15 13:02:26 hewitson Exp $
%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
classdef pzmodel < ltpda_tf
  
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  %                            Property definition                            %
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  
  %---------- Public (read/write) Properties  ----------
  properties
  end
  
  %---------- Protected read-only Properties ----------
  properties (SetAccess = protected)
    poles   = []; % pole-array of the model
    zeros   = []; % zero-array of the model
    gain    = NaN; % gain of the model
    delay   = 0; % delay of the pole/zero model
  end
  
  %---------- Private Properties ----------
  properties (GetAccess = protected, SetAccess = protected)
  end
  
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  %                          Check property setting                           %
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  
  methods
    function set.gain(obj, val)
      if ~isnumeric(val) || isempty(val) || ~isreal(val)
        error('### The value for the property ''gain'' must be a real number');
      end
      obj.gain = val;
    end
    function set.poles(obj, val)
      if ~isa(val, 'pz') && ~isempty(val)
        error('### The value for the property ''poles'' must be a vector of pz objects.');
      end
      obj.poles = val;
    end
    function set.zeros(obj, val)
      if ~isa(val, 'pz') && ~isempty(val)
        error('### The value for the property ''zeros'' must be a vector of pz objects.');
      end
      obj.zeros = val;
    end
  end
  
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  %                                Constructor                                %
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  
  methods
    function obj = pzmodel(varargin)
      
      import utils.const.*
      utils.helper.msg(msg.OMNAME, 'running %s/%s', mfilename('class'), mfilename);
      
      % Collect all pzmodel objects
      [pzs, invars, rest] = utils.helper.collect_objects(varargin(:), 'pzmodel');
      
      if isempty(rest) && ~isempty(pzs)
        % Do copy constructor and return
        utils.helper.msg(msg.OPROC1, 'copy constructor');
        obj = copy(pzs, 1);
        for kk=1:numel(obj)
          obj(kk).addHistory(pzmodel.getInfo('pzmodel', 'None'), [], [], obj(kk).hist);
        end
        return
      end
      
      switch nargin
        case 0
          %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
          %%%%%%%%%%%% Zero inputs %%%%%%%%%%%%%%%
          %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
          utils.helper.msg(msg.OPROC1, 'empty constructor');
          obj.addHistory(pzmodel.getInfo('pzmodel', 'None'), plist(), [], []);
          
        case 1
          %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
          %%%%%%%%%%%% One inputs %%%%%%%%%%%%%%%
          %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
          
          if ischar(varargin{1})
            %======== Filename
            %%%%%%%%%%   pzm = pzmodel('foo.mat')   %%%%%%%%%%
            %%%%%%%%%%   pzm = pzmodel('foo.xml')   %%%%%%%%%%
            %%%%%%%%%%   pzm = pzmodel('foo.fil')  %%%%%%%%%%
            utils.helper.msg(msg.OPROC1, 'constructing from file %s', varargin{1});
            obj = fromFile(obj, varargin{1});
            
          elseif isnumeric(varargin{1})
            %%%%%%%%% pzm = pzmodel(const) %%%%%%%%%%%%%%
            
            obj = pzmodel(varargin{1}, {}, {});
            
          elseif isstruct(varargin{1})
            %%%%%%%%%%   pzm = pzmodel(struct)   %%%%%%%%%%
            utils.helper.msg(msg.OPROC1, 'constructing from struct');
            obj = fromStruct(obj, varargin{1});
            
          elseif isa(varargin{1}, 'rational')
            %%%%%%%%%%   pzm = pzmodel(rational-object)   %%%%%%%%%%
            utils.helper.msg(msg.OPROC1, 'constructing from rational');
            obj = fromRational(obj, plist('rational', varargin{1}));
            
          elseif isa(varargin{1}, 'parfrac')
            %%%%%%%%%%   pzm = pzmodel(rational-object)   %%%%%%%%%%
            utils.helper.msg(msg.OPROC1, 'constructing from parfrac');
            obj = fromParfrac(obj, plist('parfrac', varargin{1}));
            
          elseif isa(varargin{1}, 'plist')
            %%%%%%%%%%   pzm = pzmodel(plist-object)   %%%%%%%%%%
            pl = varargin{1};
            
            if pl.isparam('filename')
              utils.helper.msg(msg.OPROC1, 'constructing from file %s', pl.find('filename'));
              obj = fromFile(obj, pl);
              
            elseif pl.isparam('hostname') || pl.isparam('conn')
              utils.helper.msg(msg.OPROC1, 'constructing from repository %s', pl.find('hostname'));
              obj = obj.fromRepository(pl);
              
            elseif pl.isparam('rational')
              utils.helper.msg(msg.OPROC1, 'constructing from rational object');
              obj = fromRational(obj, pl);
              
            elseif pl.isparam('parfrac')
              utils.helper.msg(msg.OPROC1, 'constructing from parfrac');
              obj = fromParfrac(obj, pl);

            elseif pl.isparam('gain') || pl.isparam('poles') || pl.isparam('zeros')
              utils.helper.msg(msg.OPROC1, 'constructing from poles and zeros');
              obj = fromPolesAndZeros(obj, pl);
              
            elseif pl.isparam('pzmodel')
              utils.helper.msg(msg.OPROC1, 'constructing from pole/zero model');
              obj = pzmodel(find(pl, 'pzmodel'));
              
            elseif pl.isparam('built-in')
              utils.helper.msg(msg.OPROC1, 'constructing from built-in model');
              obj = fromModel(obj, pl);
              
            elseif pl.isparam('plist')
              ipl = find(pl, 'plist');
              % pass the ipl into the constructor
              obj = pzmodel(ipl);
              
            else
              obj.setObjectProperties(pl);
              obj.addHistory(pzmodel.getInfo('pzmodel', 'None'), pl, [], []);
            end
          else
            error('### Unknown single argument constructor.');
          end
          
        case 2
          %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
          %%%%%%%%%%%%% Two inputs %%%%%%%%%%%%%%%
          %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
          
          if (isa(varargin{1}, 'database') || isa(varargin{1}, 'mpipeline.repository.RepositoryConnection')) && isnumeric(varargin{2})
            %%%%%%%%%%  f = pzmodel(<database-object>, [IDs])   %%%%%%%%%%
            utils.helper.msg(msg.OPROC1, 'retrieve from repository');
            obj = obj.fromRepository(plist('conn', varargin{1}, 'id', varargin{2}));
            
          elseif isa(varargin{1}, 'pzmodel') && isa(varargin{2}, 'plist') && isempty(varargin{2}.params)
            %%%%%%%%%%  f = pzmodel(pzmodel-object, <empty plist>)   %%%%%%%%%%
            obj = pzmodel(varargin{1});
            
          elseif isa(varargin{1}, 'org.apache.xerces.dom.DeferredElementImpl') && ...
                 isa(varargin{2}, 'history')
            %%%%%%%%%%   obj = pzmodel(DOM node, history-objects)   %%%%%%%%%%
            obj = fromDom(obj, varargin{1}, varargin{2});
            
          elseif isa(varargin{1}, 'ltpda_uoh') && isa(varargin{2}, 'plist')
            %%%%%%%%%%%   pzmodel(<ltpda_uoh>-object, plist-object)   %%%%%%%%%%
            % always recreate from plist
            
            % If we are trying to load from file, and the file exists, do
            % that. Otherwise, copy the input object.
            if varargin{2}.isparam('filename')
              if exist(fullfile('.', find(varargin{2}, 'filename')), 'file')==2
                obj = pzmodel(varargin{2});
              else
                obj = pzmodel(varargin{1});
              end
            else
              obj = pzmodel(varargin{2});
            end
            
          else
            error('### Unknown 2 argument constructor.');
          end
          
        case 3
          %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
          %%%%%%%%%%%% Three inputs %%%%%%%%%%%%%%
          %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
          
          %%%%%%%%%%   pzm = pzmodel(gain, poles, zeros)   %%%%%%%%%%
          utils.helper.msg(msg.OPROC1, 'constructing from poles and zeros');
          pl  = plist('gain', varargin{1}, 'poles', varargin{2}, 'zeros', varargin{3});
          obj = fromPolesAndZeros(obj, pl);
        case 4
          %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
          %%%%%%%%%%%% Four inputs %%%%%%%%%%%%%%%
          %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
          
          if isnumeric(varargin{1})  && ischar(varargin{4})
            %%%%%%%%%%   pzm = pzmodel(gain, poles, zeros, name)   %%%%%%%%%%
            utils.helper.msg(msg.OPROC1, 'constructing from poles and zeros');
            pl  = plist('gain', varargin{1}, 'poles', varargin{2}, 'zeros', varargin{3}, 'name', varargin{4});
            obj = fromPolesAndZeros(obj, pl);
            
          elseif isnumeric(varargin{1})  && isnumeric(varargin{4})
            %%%%%%%%%%   pzm = pzmodel(gain, poles, zeros, delay)   %%%%%%%%%%
            utils.helper.msg(msg.OPROC1, 'constructing from poles and zeros and delay');
            pl  = plist('gain', varargin{1}, 'poles', varargin{2}, 'zeros', varargin{3}, 'delay', varargin{4});
            obj = fromPolesAndZeros(obj, pl);
            
          else
            error('### Unknown 4 argument constructor.');
          end
          
          
        case 5
          %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
          %%%%%%%%%%%% five inputs %%%%%%%%%%%%%%%
          %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
          
          if isnumeric(varargin{1}) && isa(varargin{4}, 'unit') && isa(varargin{5}, 'unit')
            %%%%%%%%%%   pzm = pzmodel(gain, poles, zeros, in<unit-object>, out<unit-object>)   %%%%%%%%%%
            utils.helper.msg(msg.OPROC1, 'constructing from poles, zeros and i/o-units');
            pl  = plist('gain', varargin{1}, 'poles', varargin{2}, 'zeros', varargin{3}, 'iunits', varargin{4}, 'ounits', varargin{5});
            obj = fromPolesAndZeros(obj, pl);
            
          elseif isnumeric(varargin{1})  && isnumeric(varargin{4}) && ischar(varargin{5})
            %%%%%%%%%%   pzm = pzmodel(gain, poles, zeros, delay, name)   %%%%%%%%%%
            utils.helper.msg(msg.OPROC1, 'constructing from poles, zeros delay and name');
            pl  = plist('gain', varargin{1}, 'poles', varargin{2}, 'zeros', varargin{3}, 'delay', varargin{4}, 'name', varargin{5});
            obj = fromPolesAndZeros(obj, pl);
            
          else
          end
          
        otherwise
          [pzs, invars, rest] = utils.helper.collect_objects(args, 'pzmodel');
          
          %%% Do we have a list of PZMODELs as input
          if ~isempty(pzs) && isempty(rest)
            obj = pzmodel(pzs);
          else
            error('### Unknown number of arguments.');
          end
      end
      
    end % End constructor
    
  end
  
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  %                            Methods (static)                               %
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  methods (Static)
    
    function mdls = getBuiltInModels(varargin)
      mdls = ltpda_uo.getBuiltInModels('pzmodel');
    end
    
    function out = VEROUT()
      out = '$Id: pzmodel.m,v 1.130 2011/08/15 13:02:26 hewitson Exp $';
    end
    
    function ii = getInfo(varargin)
      ii = utils.helper.generic_getInfo(varargin{:}, 'pzmodel');
    end
    
    function out = SETS()
      out = [SETS@ltpda_uoh, ...
        {'From LISO File'},  ...
        {'From Rational'},   ...
        {'From Parfrac'},   ...
        {'From Poles/Zeros'}];
    end
    
    function plout = getDefaultPlist(set)
      persistent pl;
      persistent lastset;
      if exist('pl', 'var')==0 || isempty(pl) || ~strcmp(lastset, set)
        pl = pzmodel.buildplist(set);
        lastset = set;
      end
      plout = pl;
    end
    
    function out = buildplist(set)
      
      if ~utils.helper.ismember(lower(pzmodel.SETS), lower(set))
        error('### Unknown set [%s]', set);
      end
      
      out = plist();
      out = pzmodel.addGlobalKeys(out);
      out = buildplist@ltpda_uoh(out, set);
      
      switch lower(set)
        case 'from poles/zeros'
          
          % Gain
          p = param({'gain','Model gain.'}, paramValue.DOUBLE_VALUE(1));
          out.append(p);
          
          % Poles
          p = param({'poles',['Vector/Cell-array of poles. Use either pz objects or the format<br>'...
            'like <tt>{[f1, q1], f2, f3, [f4, q4]}<tt>']}, paramValue.EMPTY_DOUBLE);
          out.append(p);
          
          % Zeros
          p = param({'zeros',['Vector/Cell-array of zeros. Use either pz objects or the format<br>'...
            'like <tt>{[f1, q1], f2, f3, [f4, q4]}<tt>']}, paramValue.EMPTY_DOUBLE);
          out.append(p);
          
          % Iunits
          p = param({'iunits','The input units of the model.'}, paramValue.EMPTY_STRING);
          out.append(p);
          
          % Ounits
          p = param({'ounits','The output units of the model.'}, paramValue.EMPTY_STRING);
          out.append(p);
          
          % Delay
          p = param({'delay','The delay of the model in seconds.'}, paramValue.DOUBLE_VALUE(0));
          out.append(p);

        case 'from rational'
          
          % rational
          p = param({'rational','Rational transfer-function model object to design from.'}, {1, {rational}, paramValue.OPTIONAL});
          out.append(p);
          
          % Iunits
          p = param({'iunits','The input units of the model.'}, paramValue.EMPTY_STRING);
          out.append(p);
          
          % Ounits
          p = param({'ounits','The output units of the model.'}, paramValue.EMPTY_STRING);
          out.append(p);
          
        case 'from parfrac'
          
          % rational
          p = param({'parfrac','Partial fractions transfer-function model object to design from.'}, {1, {parfrac}, paramValue.OPTIONAL});
          out.append(p);
          
          % Iunits
          p = param({'iunits','The input units of the model.'}, paramValue.EMPTY_STRING);
          out.append(p);
          
          % Ounits
          p = param({'ounits','The output units of the model.'}, paramValue.EMPTY_STRING);
          out.append(p);
          
        case 'from liso file'
          % Filename
          p = param({'filename','LISO filename.'}, paramValue.EMPTY_STRING);
          out.append(p);
      end
    end % function out = getDefaultPlist(varargin)
    
    function obj = initObjectWithSize(n,m)
      obj = pzmodel.newarray([n m]);
    end
    
  end % End static methods
  
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  %                         Methods (static, private)                         %
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  
  methods (Static, Access=private)
    
    [ao,bo] = abcascade(a1,b1,a2,b2)
    
  end % End static, private methods
  
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  %                         Methods (static, hidden)                          % 
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  
  methods (Static = true, Hidden = true)
    varargout = loadobj(varargin)
    varargout = update_struct(varargin);
  end
  
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  %                             Methods (public)                              %
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  methods
    varargout = char(varargin)
    varargout = display(varargin)
    varargout = copy(varargin)
    
    f = getlowerFreq(varargin)
    f = getupperFreq(varargin)
    
    varargout = tomiir(varargin)
    varargout = tomfir(varargin)
    varargout = fngen(varargin)

    varargout = setGain(varargin)
    varargout = setDelay(varargin)
    varargout = setPoles(varargin)
    varargout = setZeros(varargin)
  end
  
  methods (Hidden = true)
    varargout = attachToDom(varargin)
  end
  
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  %                            Methods (protected)                            %
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  methods (Access = protected)
    varargout = fromLISO(varargin)
    varargout = fromStruct(varargin)
    varargout = fromDom(varargin)
    varargout = pzm2ab(varargin)
    varargout = respCore(varargin)
  end
  
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  %                           Methods (private)                               %
  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
  methods (Access = private)
    % Constructors
    varargout = fromPolesAndZeros(varargin)
    varargout = fromRational(varargin)
    varargout = fromParfrac(varargin)
  end
  
end % End classdef