| 1 | % 'xml2struct': read an xml file as a Matlab structure, converts numeric character strings into numbers |
|---|
| 2 | %----------------------------------------------------------------------- |
|---|
| 3 | % function s=xml2struct(filename) |
|---|
| 4 | % |
|---|
| 5 | % OUTPUT: |
|---|
| 6 | % s= Matlab structure corresponding to the input xml file |
|---|
| 7 | % |
|---|
| 8 | % INPUT: |
|---|
| 9 | % filename: name of the xml file |
|---|
| 10 | % varargin: optional list of strings to restrict the reading to a selection of subtrees, for instance 'GeometryCalib' (save time) |
|---|
| 11 | |
|---|
| 12 | function [s,Heading]=xml2struct(filename,varargin) |
|---|
| 13 | t=xmltree(filename); |
|---|
| 14 | iline=0; |
|---|
| 15 | Heading=''; |
|---|
| 16 | while isempty(Heading) |
|---|
| 17 | iline=iline+1; |
|---|
| 18 | if strcmp(get(t,iline,'type'),'element') |
|---|
| 19 | Heading=get(t,iline,'name'); |
|---|
| 20 | end |
|---|
| 21 | end |
|---|
| 22 | if nargin>1 |
|---|
| 23 | for isub=1:nargin-1 |
|---|
| 24 | uid_sub=find(t,['/' Heading '/' varargin{isub}]); |
|---|
| 25 | if isempty(uid_sub) |
|---|
| 26 | s.(varargin{isub})=[]; |
|---|
| 27 | else |
|---|
| 28 | tsub=branch(t,uid_sub); |
|---|
| 29 | ss=convert(tsub); |
|---|
| 30 | s.(varargin{isub})=convert_string(ss); |
|---|
| 31 | end |
|---|
| 32 | end |
|---|
| 33 | else |
|---|
| 34 | ss=convert(t); |
|---|
| 35 | s=convert_string(ss); |
|---|
| 36 | end |
|---|
| 37 | |
|---|
| 38 | |
|---|
| 39 | function out=convert_string(ss) |
|---|
| 40 | info=whos('ss'); |
|---|
| 41 | switch info.class |
|---|
| 42 | case 'struct' |
|---|
| 43 | out=[];%default |
|---|
| 44 | names = fieldnames(ss); |
|---|
| 45 | for k=1:length(names) |
|---|
| 46 | out.(names{k})=convert_string(ss.(names{k})); |
|---|
| 47 | end |
|---|
| 48 | case 'char' |
|---|
| 49 | if isempty(regexp(ss,'^(-*\d+\.*\d*\ *)+$'))% if the string does not contains a set of numbers (with possible sign and decimal) separated by blanks |
|---|
| 50 | sep_ind=regexp(ss,'\s&\s');% check for separator ' & ' which indicates column separation in tables |
|---|
| 51 | if ~isempty(sep_ind) |
|---|
| 52 | sep_ind=[-2 sep_ind length(ss)+1]; |
|---|
| 53 | for icolumn=1:length(sep_ind)-1 |
|---|
| 54 | out{1,icolumn}=ss(sep_ind(icolumn)+3:sep_ind(icolumn+1)-1); |
|---|
| 55 | end |
|---|
| 56 | else |
|---|
| 57 | out=ss; %reproduce the input string |
|---|
| 58 | end |
|---|
| 59 | else |
|---|
| 60 | out=str2num(ss); |
|---|
| 61 | end |
|---|
| 62 | case 'cell' |
|---|
| 63 | out=[];%default |
|---|
| 64 | check_numeric=zeros(size(ss)); |
|---|
| 65 | for ilist=1:numel(ss) |
|---|
| 66 | if ~isempty(str2num(ss{ilist})) |
|---|
| 67 | out{ilist,1}=str2num(ss{ilist}); |
|---|
| 68 | check_numeric(ilist)=1; |
|---|
| 69 | else |
|---|
| 70 | sep_ind=regexp(ss{ilist},'\s&\s');% check for separator ' & ' which indicates column separation in tables |
|---|
| 71 | if ~isempty(sep_ind) |
|---|
| 72 | sep_ind=[-2 sep_ind length(ss{ilist})+1]; |
|---|
| 73 | for icolumn=1:length(sep_ind)-1 |
|---|
| 74 | out{ilist,icolumn}=ss{ilist}(sep_ind(icolumn)+3:sep_ind(icolumn+1)-1); |
|---|
| 75 | end |
|---|
| 76 | else |
|---|
| 77 | out{ilist,1}=ss{ilist}; %reproduce the input string |
|---|
| 78 | end |
|---|
| 79 | end |
|---|
| 80 | end |
|---|
| 81 | if isequal(check_numeric,ones(size(ss))) |
|---|
| 82 | out=cell2mat(out); |
|---|
| 83 | end |
|---|
| 84 | otherwise |
|---|
| 85 | out=ss; |
|---|
| 86 | end |
|---|
| 87 | |
|---|
| 88 | |
|---|