source: trunk/src/xml2struct.m

Last change on this file was 1150, checked in by sommeria, 3 months ago

browse_data improved

File size: 4.8 KB
Line 
1% 'xml2struct': read an xml file as a Matlab structure, converts numeric character strings into numbers
2%-----------------------------------------------------------------------
3% function [s,RootTag,errormsg]=xml2struct(filename,varargin)
4%
5% OUTPUT:
6% s= Matlab structure corresponding to the input xml file
7% RootTag= name of the root tag in the xml file
8% errormsg: errormessage, ='' by default
9%
10% INPUT:
11% filename: name of the xml file
12% varargin: optional list of strings to restrict the reading to a selection of subtrees, for instance 'GeometryCalib' (save reading time)
13
14%=======================================================================
15% Copyright 2008-2024, LEGI UMR 5519 / CNRS UGA G-INP, Grenoble, France
16
17%   http://www.legi.grenoble-inp.fr
18%   Joel.Sommeria - Joel.Sommeria (A) univ-grenoble-alpes.fr
19%
20%     This file is part of the toolbox UVMAT.
21%
22%     UVMAT is free software; you can redistribute it and/or modify
23%     it under the terms of the GNU General Public License as published
24%     by the Free Software Foundation; either version 2 of the license,
25%     or (at your option) any later version.
26%
27%     UVMAT is distributed in the hope that it will be useful,
28%     but WITHOUT ANY WARRANTY; without even the implied warranty of
29%     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
30%     GNU General Public License (see LICENSE.txt) for more details.
31%=======================================================================UVMAT-https     
32
33function [s,RootTag,errormsg]=xml2struct(filename,varargin)
34s=[];
35RootTag='';
36errormsg='';
37try
38    t=xmltree(filename);% read the file as an xmltree object t
39catch ME
40    errormsg=ME.message;
41    if ~isempty(regexp(ME.message,'Undefined function'))||~isempty(regexp(ME.message,'Missing'))
42        errormsg=[errormsg ': package xmltree not correctly installed, reload it from www.artefact.tk/software/matlab/xml'];
43    end
44    return
45end
46iline=0;
47
48while isempty(RootTag)
49    iline=iline+1;
50    if strcmp(get(t,iline,'type'),'element')
51        RootTag=get(t,iline,'name');
52    end
53end
54if nargin>1
55    for isub=1:nargin-1
56        uid_sub=find(t,['/' RootTag '/' varargin{isub}]);
57        if isempty(uid_sub)
58            s.(varargin{isub})=[];
59        else
60        tsub=branch(t,uid_sub);
61        if ~isempty(get(tsub,1,'contents'))
62        ss=convert(tsub);
63        s.(varargin{isub})=convert_string(ss);
64        end
65        end
66    end
67else
68    try
69    ss=convert(t);%transform the xmltree object into a Matlab structure.
70    s=convert_string(ss);
71    catch ME
72        errormsg=ME.message;
73    end
74end
75
76
77function out=convert_string(ss)
78info=whos('ss');
79switch info.class
80    case 'struct'
81        out=[];%default
82        names = fieldnames(ss);
83        for k=1:length(names)
84            out.(names{k})=convert_string(ss.(names{k}));
85        end
86    case 'char'
87        % try to convert to number if the char does not correspond to a function (otherwise str2num calls this function as it uses 'eval')
88%         if exist(ss,'builtin')||exist(ss,'file')% ss corresponds to the name of a builtin Matlab function or a file
89%             out=ss; %reproduce the input string
90%         else
91            out=str2num(ss);% convert to number or vector (str2num applied to a fct name executes this fct by 'eval', thus this possibility had to be ruled out above
92            if isempty(out)
93                sep_ind=regexp(ss,'\s&\s');% check for separator ' & ' which indicates column separation in tables
94                if ~isempty(sep_ind)
95                    sep_ind=[-2 sep_ind length(ss)+1];
96                    out={};
97                    for icolumn=1:length(sep_ind)-1
98                        out{1,icolumn}=ss(sep_ind(icolumn)+3:sep_ind(icolumn+1)-1);% get info between separators as a cell array
99                    end
100                else
101                    out=ss; %reproduce the input string
102                end
103            end
104%         end
105    case 'cell'
106        out={};%default
107        check_numeric=zeros(size(ss));
108        for ilist=1:numel(ss)
109            if ~strcmp(ss{ilist},'image') && ~isempty(str2num(ss{ilist}))
110                out{ilist,1}=str2num(ss{ilist});
111                check_numeric(ilist)=1;
112            else
113                sep_ind=regexp(ss{ilist},'\s&\s');% check for separator ' & ' which indicates column separation in tables
114                if ~isempty(sep_ind)
115                    sep_ind=[-2 sep_ind length(ss{ilist})+1];
116                    for icolumn=1:length(sep_ind)-1
117                        out{ilist,icolumn}=ss{ilist}(sep_ind(icolumn)+3:sep_ind(icolumn+1)-1);
118                    end
119                else
120                    out{ilist,1}=ss{ilist}; %reproduce the input string
121                end
122            end
123        end
124        if isequal(check_numeric,ones(size(ss)))
125            out=cell2mat(out);
126        end
127    otherwise
128        out=ss;
129end
130
131   
Note: See TracBrowser for help on using the repository browser.