source: trunk/src/series/extract_rdvision.m @ 1059

Last change on this file since 1059 was 1059, checked in by sommeria, 2 years ago

various bugs repaired

File size: 23.8 KB
Line 
1%'extract_rdvision': relabel an image series with two indices, and correct errors from the RDvision transfer program
2%------------------------------------------------------------------------
3% function ParamOut=extract_rdvision(Param)
4%------------------------------------------------------------------------
5%
6%%%%%%%%%%% GENERAL TO ALL SERIES ACTION FCTS %
7%%%%%%%%%%%%%%%%%%%%%%%%%%
8%
9%OUTPUT
10% ParamOut: sets options in the GUI series.fig needed for the function
11%
12%INPUT:
13% In run mode, the input parameters are given as a Matlab structure Param copied from the GUI series.
14% In batch mode, Param is the name of the corresponding xml file containing the same information
15% when Param.Action.RUN=0 (as activated when the current Action is selected
16% in series), the function ouput paramOut set the activation of the needed GUI elements
17%
18% Param contains the elements:(use the menu bar command 'export/GUI config' in series to
19% see the current structure Param)
20%    .InputTable: cell of input file names, (several lines for multiple input)
21%                      each line decomposed as {RootPath,SubDir,Rootfile,NomType,Extension}
22%    .OutputSubDir: name of the subdirectory for data outputs
23%    .OutputDirExt: directory extension for data outputs
24%    .Action: .ActionName: name of the current activated function
25%             .ActionPath:   path of the current activated function
26%             .ActionExt: fct extension ('.m', Matlab fct, '.sh', compiled   Matlab fct
27%             .RUN =0 for GUI input, =1 for function activation
28%             .RunMode='local','background', 'cluster': type of function  use
29%             
30%    .IndexRange: set the file or frame indices on which the action must be performed
31%    .FieldTransform: .TransformName: name of the selected transform function
32%                     .TransformPath:   path  of the selected transform function
33%    .InputFields: sub structure describing the input fields withfields
34%              .FieldName: name(s) of the field
35%              .VelType: velocity type
36%              .FieldName_1: name of the second field in case of two input series
37%              .VelType_1: velocity type of the second field in case of two input series
38%              .Coord_y: name of y coordinate variable
39%              .Coord_x: name of x coordinate variable
40%    .ProjObject: %sub structure describing a projection object (read from ancillary GUI set_object)
41%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
42
43%=======================================================================
44% Copyright 2008-2018, LEGI UMR 5519 / CNRS UGA G-INP, Grenoble, France
45%   http://www.legi.grenoble-inp.fr
46%   Joel.Sommeria - Joel.Sommeria (A) legi.cnrs.fr
47%
48%     This file is part of the toolbox UVMAT.
49%
50%     UVMAT is free software; you can redistribute it and/or modify
51%     it under the terms of the GNU General Public License as published
52%     by the Free Software Foundation; either version 2 of the license,
53%     or (at your option) any later version.
54%
55%     UVMAT is distributed in the hope that it will be useful,
56%     but WITHOUT ANY WARRANTY; without even the implied warranty of
57%     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
58%     GNU General Public License (see LICENSE.txt) for more details.
59%=======================================================================
60
61function ParamOut=extract_rdvision(Param) %default output=relabel_i_j(Param)
62
63%% set the input elements needed on the GUI series when the action is selected in the menu ActionName
64if isstruct(Param) && isequal(Param.Action.RUN,0)
65    ParamOut.AllowInputSort='off';...% allow alphabetic sorting of the list of input file SubDir (options 'off'/'on', 'off' by default)
66        ParamOut.WholeIndexRange='on';...% prescribes the file index ranges from min to max (options 'off'/'on', 'off' by default)
67        ParamOut.NbSlice=1; ...%nbre of slices, 1 prevents splitting in several processes, ('off' by default)
68        ParamOut.VelType='off';...% menu for selecting the velocity type (options 'off'/'one'/'two',  'off' by default)
69        ParamOut.FieldName='off';...% menu for selecting the field (s) in the input file(options 'off'/'one'/'two', 'off' by default)
70        ParamOut.FieldTransform = 'off';...%can use a transform function
71        ParamOut.ProjObject='off';...%can use projection object(option 'off'/'on',
72        ParamOut.Mask='off';...%can use mask option   (option 'off'/'on', 'off' by default)
73        ParamOut.OutputDirExt='.extract';%set the output dir extension
74    ParamOut.OutputSubDirMode='one'; %output folder given by the folder name of the first input line
75    % detect the set of image folder
76    RootPath=Param.InputTable{1,1};
77    ListStruct=dir(RootPath);
78    ListCells=struct2cell(ListStruct);% transform dir struct to a cell arrray
79    check_bad=strcmp('.',ListCells(1,:))|strcmp('..',ListCells(1,:));%detect the dir '.' to exclude it
80    check_dir=cell2mat(ListCells(4,:));% =1 for directories, =0 for files
81    ListDir=ListCells(1,find(check_dir & ~check_bad));
82    %     InputTable=cell(numel(ListDir),5);
83    %     InputTable(:,2)=ListDir';
84    isel=0;
85    InputTable=Param.InputTable;
86    for ilist=1:numel(ListDir)
87        ListStructSub=dir(fullfile(RootPath,ListDir{ilist}));
88        ListCellSub=struct2cell(ListStructSub);% transform dir struct to a cell arrray
89        detect_seq=regexp(ListCellSub(1,:),'.seq$');
90        seq_index=find(~cellfun('isempty',detect_seq),1);
91        if ~isempty(seq_index)
92            %             msgbox_uvmat('ERROR',['not seq file in ' ListDir{ilist} ': please check the input folders'])
93            %         else
94            isel=isel+1;
95            InputTable{isel,1}=RootPath;
96            InputTable{isel,2}=ListDir{ilist};
97            RootFile=regexprep(ListCellSub{1,seq_index},'.seq$','');
98            InputTable{isel,3}=RootFile;
99            InputTable{isel,4}='*';
100            InputTable{isel,5}='.seq';
101        end
102    end
103    hseries=findobj(allchild(0),'Tag','series');% find the parent GUI 'series'
104    hhseries=guidata(hseries); %handles of the elements in 'series'
105    set(hhseries.InputTable,'Data',InputTable)
106    ParamOut.ActionInput.LogPath=RootPath;% indicate the path for the output info: 0_LOG ....
107    return
108end
109
110ParamOut=[];
111%%%%%%%%%%%% STANDARD PART  %%%%%%%%%%%%
112%% read input parameters from an xml file if input is a file name (batch mode)
113
114if ischar(Param)
115    Param=xml2struct(Param);% read Param as input file (batch case)
116%     checkrun=0;
117end
118disp(Param)
119checkrun=strcmp(Param.RunMode,'local')
120hseries=findobj(allchild(0),'Tag','series');
121RUNHandle=findobj(hseries,'Tag','RUN');%handle of RUN button in GUI series
122WaitbarHandle=findobj(hseries,'Tag','Waitbar');%handle of waitbar in GUI series
123
124%% root input file(s) and type
125RootPath=Param.InputTable{1,1};
126if ~isempty(find(~strcmp(RootPath,Param.InputTable(:,1))))% if the Rootpath for each camera are not identical
127    disp_uvmat('ERROR','Rootpath for all cameras must be identical',checkrun)
128    return
129end
130
131% get the set of input file names (cell array filecell), and the lists of
132% input file or frame indices i1_series,i2_series,j1_series,j2_series
133[filecell,i1_series,i2_series,j1_series,j2_series]=get_file_series(Param);
134
135%OutputDir=[Param.OutputSubDir Param.OutputDirExt];
136 
137% numbers of slices and file indices
138nbfield_j=size(i1_series{1},1); %nb of fields for the j index (bursts or volume slices)
139nbfield_i=size(i1_series{1},2); %nb of fields for the i index
140nbfield=nbfield_j*nbfield_i; %total number of fields
141
142%determine the file type on each line from the first input file
143
144FileInfo=get_file_info(filecell{1,1});
145if strcmp(FileInfo.FileType,'rdvision')
146    if ~isequal(FileInfo.NumberOfFrames,nbfield)
147        disp_uvmat('WARNING',['the whole series of ' num2str(FileInfo.NumberOfFrames) ' images must be extracted at once'],checkrun)
148        %rmfield(OutputDir)
149%         return
150    end
151    %% interactive input of specific parameters (for RDvision system)
152    display('converting images from RDvision system...')
153else
154    disp_uvmat('ERROR','the input is not from rdvision: a .seq or .sqb file must be opened',checkrun)
155    return
156end
157t=xmltree;
158
159save(t,fullfile(RootPath,'Running.xml'))%create an xml file to indicate that processing takes place
160
161%% calibration data and timing: read the ImaDoc files
162mode=''; %default
163timecell={};
164itime=0;
165NbSlice_calib={};
166
167%SubDirBase=regexprep(SubDir{1},'\..*','');%take the root part of SubDir, before the first dot '.'
168
169%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
170%%%  loop on the cameras ( #iview)
171%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
172% RootPath=Param.InputTable(:,1);
173% RootFile=Param.InputTable(:,3);
174% SubDir=Param.InputTable(:,2);
175% NomType=Param.InputTable(:,4);
176% FileExt=Param.InputTable(:,5);
177
178% [XmlData,NbSlice_calib,time,errormsg]=read_multimadoc(RootPath,SubDir,RootFile,FileExt,i1_series,i2_series,j1_series,j2_series);
179% if size(time,1)>1
180%     diff_time=max(max(diff(time)));
181%     if diff_time>0
182%         disp_uvmat('WARNING',['times of series differ by (max) ' num2str(diff_time)],checkrun)
183%     end
184% end
185%
186%      nbfield2=size(time,1);
187checkpreserve=0;% if =1, will npreserve the original images, else it erases them at the end
188
189for iview=1:size(Param.InputTable,1)
190    check_xml=0;
191    for iview_xml=1:size(Param.InputTable,1)% look for the xml files in the different data directories
192        filexml=[fullfile(RootPath,Param.InputTable{iview_xml,2},Param.InputTable{iview,3}) '.xml'];%new convention: xml at the level of the image folder
193        if exist(filexml,'file')
194            check_xml=1;
195            break
196        end
197    end
198    %     if ~exist(filexml,'file')
199    %         disp_uvmat('ERROR',[filexml ' missing'],checkrun)
200    %         return
201    %     end
202   
203    newxml=fullfile(RootPath,Param.InputTable{iview,3});
204    newxml=regexprep(newxml,'_Master_Dalsa_4M180$','');%suppress '_Master_Dalsa_4M180'
205    newxml=[newxml '.xml'];
206   
207    %% get the names of .seq and .sqb files
208    switch Param.InputTable{iview,5}
209        case {'.seq','.sqb'}
210            filename_seq=fullfile(RootPath,Param.InputTable{iview,2},[Param.InputTable{iview,3} '.seq']);
211            filename_sqb=fullfile(RootPath,Param.InputTable{iview,2},[Param.InputTable{iview,3} '.sqb']);
212           
213            logdir=[Param.OutputSubDir Param.OutputDirExt];
214            [success,errormsg] = copyfile(filename_seq,[fullfile(RootPath,logdir,Param.InputTable{iview,3}) '.seq']); %copy the seq file in the upper folder
215            [success,errormsg] = copyfile(filename_sqb,[fullfile(RootPath,logdir,Param.InputTable{iview,3}) '.sqb']); %copy the sqb file in the upper folder
216            if check_xml
217            [success,errormsg] = copyfile(filexml,[fullfile(RootPath,logdir,Param.InputTable{iview,3}) '.xml']); %copy the original xml file in the upper folder
218            end
219        otherwise
220            errormsg='input file extension must be .seq or .sqb';
221    end
222    if ~exist(filename_seq,'file')
223        errormsg=[filename_seq ' does not exist'];
224    end
225    if ~isempty(errormsg)
226        disp_uvmat('ERRROR',errormsg,checkrun);
227        return
228    end
229   
230
231    %% get data from .seq file
232    s=ini2struct(filename_seq);
233    SeqData=s.sequenceSettings;
234    SeqData.width=str2double(SeqData.width);
235    SeqData.height=str2double(SeqData.height);
236    SeqData.bytesperpixel=str2double(SeqData.bytesperpixel);
237    SeqData.nb_frames=str2double(s.sequenceSettings.numberoffiles);
238    if isempty(SeqData.binrepertoire)%used when binrepertoire empty, strange feature of rdvision
239        SeqData.binrepertoire=regexprep(s.sequenceSettings.bindirectory,'\\$','');%tranform Windows notation to Linux
240        SeqData.binrepertoire=regexprep(SeqData.binrepertoire,'\','/');
241        [tild,SeqData.binrepertoire,DirExt]=fileparts(SeqData.binrepertoire);
242    end
243   
244   
245   
246    %% reading the .sqb file
247    m = memmapfile(filename_sqb,'Format', { 'uint32' [1 1] 'offset'; ...
248        'uint32' [1 1] 'garbage1';...
249        'double' [1 1] 'timestamp';...
250        'uint32' [1 1] 'file_idx';...
251        'uint32' [1 1] 'garbage2' },'Repeat',SeqData.nb_frames);
252   
253    %%%%%%%BRICOLAGE in case of unreadable .sqb file: remplace lecture du fichier
254%             ind=[8356 8356:8672];%indices of bin files
255%             w=2432;%w=width of images in pixels
256%             h=864;%h=height of images in pixels
257%             bpp=2;% nbre of bytes per pixel
258%             lengthimage=w*h*bpp;% lengthof an image record on the binary file
259%             nbimages=15; %nbre of images of each camera in a bin file
260%             for ii=1:15*numel(ind)
261%                 %data(ii).offset=mod(ii-1,32)*2*lengthimage+lengthimage;%Dalsa_2
262%                 %data(ii).offset=mod(ii-1,32)*2*lengthimage;%Dalsa_1
263%                 %data(ii).file_idx=ind(ceil(ii/32));
264%                 data(ii).file_idx=ind(ceil(ii/15));
265%                 data(ii).timestamp=0.005*(ii-1);
266%             end
267%             m.Data=data;
268    %%%%%%%
269    timestamp=zeros(1,numel(m.Data));
270    for ii=1: numel(m.Data)
271        timestamp(ii)=m.Data(ii).timestamp;
272    end
273    [nbfield1,nbfield2,msg]=copyfile_modif(filexml,timestamp,newxml); %copy the xml file in the upper folder
274    [XmlData,errormsg]=imadoc2struct(newxml);% check reading of the new xml file
275    if ~isempty(errormsg)
276        disp(errormsg)
277        return
278    end
279    timestamp=timestamp(1:nbfield1*nbfield2);
280    timestamp=reshape(timestamp,nbfield2,nbfield1);
281    difftime=XmlData.Time(2:end,2:end)'-timestamp;
282    disp(['time from xml and timestamp differ by ' num2str(max(max(abs(difftime))))])
283    if max(abs(difftime))>0.01
284        checkpreserve=1;% will not erase the initial files, possibility of error
285    end
286   
287        %% checking consistency with the xml file
288%     if ~isequal(SeqData.nb_frames,numel(timestamp))
289%         disp_uvmat('ERRROR',['inconsistent number of images ' num2str(SeqData.nb_frames) ' with respect to the xml file: ' num2str(numel(timestamp))] ,checkrun);
290%         return
291%     end   
292   
293    if nbfield2>1
294        NomTypeNew='_1_1';
295    else
296        NomTypeNew='_1';
297    end
298
299    [BinList,errormsg]=binread_rdv_series(RootPath,SeqData,m.Data,nbfield2,NomTypeNew);
300    if ~isempty(errormsg)
301        disp_uvmat('ERROR',errormsg,checkrun)
302        return
303    end
304   
305    % check the existence of the expected output image files (from the xml)
306    FileDir=SeqData.sequencename;
307     FileDir=regexprep(FileDir,'_Master_Dalsa_4M180$','');%suppress '_Master_Dalsa_4M180'
308    for i1=1:numel(timestamp)/nbfield2
309        for j1=1:nbfield2
310            OutputFile=fullfile_uvmat(RootPath,FileDir,'img','.png',NomTypeNew,i1,[],j1);% TODO: set NomTypeNew from SeqData.mode
311            try
312            A=imread(OutputFile);% check image reading (stop if error)
313            catch ME
314                disp(['checking ' OutputFile])
315                disp(ME.message)
316            end
317        end
318    end
319end
320
321%% remove binary files if transfer OK
322%     if ~checkpreserve
323%         for iview=1:size(Param.InputTable,1)
324%          fullfile(RootPath,Param.InputTable{iview,2})
325%          source_dir=fullfile(RootPath,Param.InputTable{iview,2});
326%         [SUCCESS,MESSAGE]=rmdir(source_dir,'s')
327%         end
328%     end
329delete(fullfile(RootPath,'Running.xml'))%delete the  xml file to indicate that processing is finished
330
331%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
332%--------- reads a series of bin files
333
334%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
335function [BinList,errormsg]=binread_rdv_series(PathDir,SeqData,SqbData,nbfield2,NomTypeNew)
336% BINREAD_RDV Permet de lire les fichiers bin g???n???r???s par Hiris ??? partir du
337% fichier seq associ???.
338%   [IMGS,TIMESTAMPS,NB_FRAMES] = BINREAD_RDV(FILENAME,FRAME_IDX) lit
339%   l'image d'indice FRAME_IDX de la s???quence FILENAME.
340%
341%   Entr???es
342%   -------
343%   FILENAME  : Nom du fichier s???quence (.seq).
344%   FRAME_IDX : Indice de l'image ??? lire. Si FRAME_IDX vaut -1 alors la
345%   s???quence est enti???rement lue. Si FRAME_IDX est un tableau d'indices
346%   alors toutes les images d'incides correspondant sont lues. Si FRAME_IDX
347%   est un tableau vide alors aucune image n'est lue mais le nombre
348%   d'images et tous les timestamps sont renvoy???s. Les indices commencent ???
349%   1 et se termines ??? NB_FRAMES.
350%
351%   Sorties
352%   -------
353%   IMGS        : Images de sortie.
354%   TIMESTAMPS  : Timestaps des images lues.
355%   NB_FRAMES   : Nombres d'images dans la s???quence.
356NbBinFile=0;
357BinSize=0;
358fid=0;
359errormsg='';
360BinList={};
361
362classname=sprintf('uint%d',SeqData.bytesperpixel*8);
363
364classname=['*' classname];
365BitDepth=8*SeqData.bytesperpixel;%needed to write images (8 or 16 bits)
366%%%%
367% SeqData.binrepertoire='2017-01-26T11.59.57';
368%SeqData.binrepertoire='2017-01-26T16.57.27';
369%SeqData.binrepertoire='2017-01-26T19.28.05';
370%SeqData.binrepertoire='2017-01-27T09.51.34';
371%SeqData.binrepertoire='2017-01-27T14.21.47'
372%%%%
373binrepertoire=fullfile(PathDir,SeqData.binrepertoire);
374FileDir=SeqData.sequencename;
375FileDir=regexprep(FileDir,'_Master_Dalsa_4M180$','');%suppress '_Master_Dalsa_4M180'
376OutputDir=fullfile(PathDir,FileDir);
377if ~exist(OutputDir,'dir')
378    %     errormsg=[OutputDir ' already exist, delete it first'];
379    %     return
380    % end
381    [s,errormsg]=mkdir(OutputDir);
382   
383    if s==0
384        disp(errormsg)
385        return%not able to create new image dir
386    end
387end
388bin_file_counter=0;
389for ii=1:SeqData.nb_frames
390    j1=[];
391    if ~isequal(nbfield2,1)
392        j1=mod(ii-1,nbfield2)+1;
393    end
394    i1=floor((ii-1)/nbfield2)+1;
395    OutputFile=fullfile_uvmat(PathDir,FileDir,'img','.png',NomTypeNew,i1,[],j1);% TODO: set NomTypeNew from SeqData.mode
396    fname=fullfile(binrepertoire,sprintf('%s%.5d.bin',SeqData.binfile,SqbData(ii).file_idx));
397    if exist(OutputFile,'file')% do not recreate existing image file
398        fid=0;
399    else
400        if fid==0 || ~strcmp(fname,fname_prev) % open the bin file if not in use
401            if fid~=0
402                fclose(fid);%close the previous bin file if relevant
403            end
404            [fid,msg]=fopen(fname,'rb');
405            if isequal(fid,-1)
406                errormsg=['error in opening ' fname ': ' msg];
407                return
408            else
409                disp([fname ' opened for reading'])
410                bin_file_counter=bin_file_counter+1;
411                BinList{bin_file_counter}=fname;
412            end
413            fseek(fid,SqbData(ii).offset,-1);%look at the right starting place in the bin file
414            NbBinFile=NbBinFile+1;%counter of binary files (for checking purpose)
415            BinSize(NbBinFile)=0;% strat counter for new bin file
416        else
417            fseek(fid,SqbData(ii).offset,-1);%look at the right starting place in the bin file
418        end
419        fname_prev=fname;
420        A=reshape(fread(fid,SeqData.width*SeqData.height,classname),SeqData.width,SeqData.height);%read the current image
421        A=A';
422        BinSize(NbBinFile)=BinSize(NbBinFile)+SeqData.width*SeqData.height*SeqData.bytesperpixel*8; %record bits read
423        try
424            imwrite(A,OutputFile,'BitDepth',BitDepth) % case of 16 bit images
425            disp([OutputFile ' written']);
426            % [s,errormsg] = fileattrib(OutputFile,'-w','a'); %set images to read only '-w' for all users ('a')
427            %         if ~s
428            % %             disp_uvmat('ERROR',errormsg,checkrun);
429            %             return
430            %         end
431        catch ME
432            errormsg=ME.message;
433            return
434        end
435    end
436end
437if fid~=0
438fclose(fid)
439end
440
441
442
443
444function [nbfield1,nbfield2,msg]=copyfile_modif(filexml,timestamp,newxml)
445msg='';
446t=xmltree(filexml);
447
448%% read Dtk and NbDtk
449NbDtk=1; %default
450Dtk=[]; % default
451uid_Dtk=find(t,'ImaDoc/Camera/BurstTiming/Dtk');
452uid_content_Dtk=get(t,uid_Dtk,'contents');
453if ~isempty(uid_content_Dtk)
454    Dtk=str2num(get(t,uid_content_Dtk,'value'));
455    uid_NbDtk=find(t,'ImaDoc/Camera/BurstTiming/NbDtk');
456    uid_content_NbDtk=get(t,uid_NbDtk,'contents');
457    if ~isempty(uid_content_NbDtk)
458    NbDtk=str2num(get(t,uid_content_NbDtk,'value'));
459    end
460end
461
462%% read Dti and NbDti
463NbDti=1; %default
464Dti=[]; % default
465uid_Dti=find(t,'ImaDoc/Camera/BurstTiming/Dti');
466uid_content_Dti=get(t,uid_Dti,'contents');
467if ~isempty(uid_content_Dti)
468    Dti=str2num(get(t,uid_content_Dti,'value'));
469    uid_NbDti=find(t,'ImaDoc/Camera/BurstTiming/NbDti');
470    uid_content_NbDti=get(t,uid_NbDti,'contents');
471    if ~isempty(uid_content_NbDti)
472    NbDti=str2num(get(t,uid_content_NbDti,'value'));
473    end
474end
475
476%% read Dtj and NbDtj
477NbDtj=1; %default
478Dtj=[]; % default
479uid_Dtj=find(t,'ImaDoc/Camera/BurstTiming/Dtj');
480uid_content_Dtj=get(t,uid_Dtj,'contents');
481if ~isempty(uid_content_Dtj)
482    Dtj=str2num(get(t,uid_content_Dtj,'value'));
483    uid_NbDtj=find(t,'ImaDoc/Camera/BurstTiming/NbDtj');
484    uid_content_NbDtj=get(t,uid_NbDtj,'contents');
485    if ~isempty(uid_content_NbDtj)
486    NbDtj=str2num(get(t,uid_content_NbDtj,'value'));
487    end
488end
489
490%% correct NbDtj and NbDti (error from RDvision)
491if NbDtj==numel(Dtj)% case of bursts
492    NbDtj=1;
493    uid_motor_nbslice=find(t,'ImaDoc/TranslationMotor/Nbslice');
494    if ~isempty(uid_motor_nbslice)&& ~isempty(uid_Dtk)% case of multilevel
495        NbSlice=str2num(get(t,get(t,uid_motor_nbslice,'contents'),'value'));
496        NbDti=NbSlice-1;
497    end
498end
499
500if isempty(Dtj)% case of simple series
501    timestamp=timestamp';
502    t=set(t,uid_content_NbDti,'value',num2str(numel(timestamp)-1));% correct NbDti in the xml file
503    nbfield1=numel(timestamp);
504    nbfield2=1;
505else
506    nbfieldi=(NbDti*numel(Dti)+1);
507    nbfieldk=(NbDtk*numel(Dtk)+1);
508    nbfield1=nbfieldi*nbfieldk;
509    nbfield2=NbDtj*numel(Dtj)+1;
510    NbFrames_xml=nbfield1*nbfield2;
511   if NbFrames_xml<numel(timestamp)
512       disp(['ERROR: size from xml ' num2str(NbFrame_xml) ' smaller than timestamp size ' num2str(numel(timestamp))])
513       return
514   end
515   if NbFrames_xml>numel(timestamp)
516       nbfield1=floor(numel(timestamp)/nbfield2);
517       nbfieldk=floor(nbfield1/nbfieldi);
518       nbfield1=nbfieldi*nbfieldk;
519       NbDtk=nbfieldk-1;
520       t=set(t,uid_content_NbDtk,'value',num2str(NbDtk));% correct NbDtk in the xml file (in practice numel(Dtk)=1;
521       timestamp=timestamp(1:nbfield1*nbfield2);
522       disp(['image record stopped before end: max index i= ' num2str(nbfield1)]);
523       timestamp=reshape(timestamp,nbfield2,nbfield1);
524   end
525   % check Dtj with respect to timestamp
526    timestamp=(reshape(timestamp,nbfield2,[]))';
527    diff_Dtj=diff(timestamp(1,:))-Dtj;
528    if max(abs(diff_Dtj))>min(Dtj)/1000
529        disp(['Dtj from xml file differs from time stamp by ' num2str(max(abs(diff_Dtj))) ', '])%'
530    else
531        disp('Dtj OK');
532    end
533end
534
535%% adjust Dti
536if NbDti+1>size(timestamp,1)
537    NbDti=size(timestamp,1)-1;
538end
539Dti_stamp=(timestamp(1+NbDti,1)-timestamp(1,1))/NbDti;
540t=set(t,uid_content_Dti,'value',num2str(Dti_stamp));%corret Dti
541if abs(Dti_stamp-Dti)>Dti/1000
542    disp([msg 'Dti from xml file corrected by ' num2str(Dti_stamp-Dti) ', ']);%'
543else
544    disp('Dti OK')
545end
546
547%% adjust Dtk
548if ~isempty(uid_Dtk)
549    Dtk_stamp=(timestamp((NbDti+1)*NbDtk+1,1)-timestamp(1,1))/NbDtk;
550    t=set(t,uid_content_Dtk,'value',num2str(Dtk_stamp));
551    if abs(Dtk_stamp-Dtk)>Dtk/1000
552        disp(['Dtk from xml file corrected by ' num2str(Dtk_stamp-Dtk)]);
553    else
554        disp('Dtk OK')
555    end
556    t=set(t,uid_content_Dtk,'value',num2str(Dtk_stamp));
557end
558
559%% save the new xml file
560save(t,newxml)
561[success,errormsg] = fileattrib(newxml,'+w','g');% allow writing access for the group of users
562if success==0
563    disp({['warning: unable to set group write access to ' newxml ':']; errormsg});%error message for directory creation
564    msg=errormsg;
565end
566
567
568
Note: See TracBrowser for help on using the repository browser.