source: trunk/src/calc_field_tps.m @ 1157

Last change on this file since 1157 was 1139, checked in by sommeria, 7 months ago

series/test patch added + some bug corrections

File size: 8.3 KB
RevLine 
[515]1
[575]2%'calc_field_tps': defines fields (velocity, vort, div...) from civ data and calculate them with tps interpolation
[515]3%---------------------------------------------------------------------
[651]4% [DataOut,VarAttribute,errormsg]=calc_field_tps(Coord_tps,NbCentre,SubRange,FieldVar,FieldName,Coord_interp)
[515]5%
6% OUTPUT:
7% DataOut: structure representing the output fields
[675]8% VarAttribute: cell array of structures coontaining the variable attributes
9% errormsg: error msg , = '' by default
[515]10%
11% INPUT:
[651]12% Coord_tps: coordinates of the centres, of dimensions [nb_point,nb_coord,nb_subdomain], where
13%            nb_point is the max number of data point in a subdomain,
14%            nb_coord the space dimension,
15%            nb_subdomain the nbre of subdomains used for tps
16% NbCentre: nbre of tps centres for each subdomain, of dimension nb_subdomain
17% SubRange: coordinate range for each subdomain, of dimensions [nb_coord,2,nb_subdomain]
[675]18% FieldVar: array representing the input fields as tps weights with dimension (nbvec_sub+3,NbSubDomain,nb_dim)
19%              nbvec_sub= max nbre of vectors in a subdomain 
20%             NbSubDomain =nbre of subdomains
21%             nb_dim: nbre of dimensions for vector components (x-> 1, y->2)
[651]22% FieldName: cell array representing the list of operations (eg div(U,V), rot(U,V))
23% Coord_interp: coordinates of sites on which the fields need to be calculated of dimensions
24%            [nb_site,nb_coord] for an array of interpolation sites
25%            [nb_site_y,nb_site_x,nb_coord] for interpolation on a plane grid of size [nb_site_y,nb_site_x]
[515]26
[809]27%=======================================================================
[1126]28% Copyright 2008-2024, LEGI UMR 5519 / CNRS UGA G-INP, Grenoble, France
[809]29%   http://www.legi.grenoble-inp.fr
[1127]30%   Joel.Sommeria - Joel.Sommeria (A) univ-grenoble-alpes.fr
[809]31%
32%     This file is part of the toolbox UVMAT.
33%
34%     UVMAT is free software; you can redistribute it and/or modify
35%     it under the terms of the GNU General Public License as published
36%     by the Free Software Foundation; either version 2 of the license,
37%     or (at your option) any later version.
38%
39%     UVMAT is distributed in the hope that it will be useful,
40%     but WITHOUT ANY WARRANTY; without even the implied warranty of
41%     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
42%     GNU General Public License (see LICENSE.txt) for more details.
43%=======================================================================
44
[651]45function [DataOut,VarAttribute,errormsg]=calc_field_tps(Coord_tps,NbCentre,SubRange,FieldVar,FieldName,Coord_interp)
[515]46
47%list of defined scalars to display in menus (in addition to 'ima_cor').
48% a type is associated to each scalar:
49%              'discrete': related to the individual velocity vectors, not interpolated by patch
50%              'vel': calculated from velocity components, continuous field (interpolated with velocity)
51%              'der': needs spatial derivatives
52%              'var': the scalar name corresponds to a field name in the netcdf files
53% a specific variable name for civ1 and civ2 fields are also associated, if
54% the scalar is calculated from other fields, as explicited below
55errormsg='';
56   
57%% nbre of subdomains
58if ndims(Coord_interp)==3
59    nb_coord=size(Coord_interp,3);
60    npx=size(Coord_interp,2);
61    npy=size(Coord_interp,1);
62    nb_sites=npx*npy;
63    Coord_interp=reshape(Coord_interp,nb_sites,nb_coord);
64else
65    nb_coord=size(Coord_interp,2);
66    nb_sites=size(Coord_interp,1);
67end
68NbSubDomain=size(Coord_tps,3);
69nbval=zeros(nb_sites,1);
70
71%% list of operations
72check_grid=0;
73check_der=0;
[675]74check_vec=0;
75check_remove=false(size(FieldName));
76VarAttribute={};
[581]77for ilist=1:length(FieldName)
[675]78    FieldNameType=regexprep(FieldName{ilist},'(.+','');% detect the char string before the parenthesis
79    VarAttributeNew={};
[581]80    switch FieldNameType
[515]81        case 'vec'
[521]82            check_grid=1;
[515]83            DataOut.U=zeros(nb_sites,1);
84            DataOut.V=zeros(nb_sites,1);
[675]85            VarAttributeNew{1}.Role='vector_x';
86            VarAttributeNew{2}.Role='vector_y';
87            check_vec=1;
88        case {'U','V'}
89            if check_vec% no new data needed
90                check_remove(ilist)=1;
91            else
[515]92            check_grid=1;
[581]93            DataOut.(FieldNameType)=zeros(nb_sites,1);
[675]94            VarAttributeNew{1}.Role='scalar';
95            end
96        case 'norm'
97            check_grid=1;
98            DataOut.(FieldNameType)=zeros(nb_sites,1);
99            VarAttributeNew{1}.Role='scalar';
[1090]100        case {'curl','div','strain','DUDX','DUDY','DVDX','DVDY'}
[515]101            check_der=1;
[581]102            DataOut.(FieldNameType)=zeros(nb_sites,1);
[675]103            VarAttributeNew{1}.Role='scalar';
[515]104    end
[675]105    VarAttribute=[VarAttribute VarAttributeNew];
[515]106end
[675]107FieldName(check_remove)=[];
[515]108
109%% loop on subdomains
[1137]110NbSubDomain
[515]111for isub=1:NbSubDomain
[651]112    nbvec_sub=NbCentre(isub);
[515]113    check_range=(Coord_interp >=ones(nb_sites,1)*SubRange(:,1,isub)' & Coord_interp<=ones(nb_sites,1)*SubRange(:,2,isub)');
[1137]114    ind_sel=find(sum(check_range,2)==nb_coord);% select points whose all coordinates are in the prescribed range   
115    x_width=(SubRange(1,2,isub)-SubRange(1,1,isub))/pi;%width of the subdomain/pi
116    y_width=(SubRange(2,2,isub)-SubRange(2,1,isub))/pi;%width of the subdomain/pi
117    CentreX=(SubRange(1,2,isub)+SubRange(1,1,isub))/2;%centre of the subdomain
118    CentreY=(SubRange(2,2,isub)+SubRange(2,1,isub))/2;
119    x_dist=(Coord_interp(ind_sel,1)-CentreX)/x_width;% relative x distance to the retangle centre*pi/2
120    y_dist=(Coord_interp(ind_sel,2)-CentreY)/y_width;% relative ydistance to the retangle centre
121    weight=cos(x_dist).*cos(y_dist);%weighting fct =1 at the rectangle center and 0 at edge
122    nbval(ind_sel)=nbval(ind_sel)+weight;% records the number of values for eacn interpolation point (in case of subdomain overlap)
[515]123    if check_grid
124        EM = tps_eval(Coord_interp(ind_sel,:),Coord_tps(1:nbvec_sub,:,isub));%kernels for calculating the velocity from tps 'sources'
125    end
126    if check_der
127        [EMDX,EMDY] = tps_eval_dxy(Coord_interp(ind_sel,:),Coord_tps(1:nbvec_sub,:,isub));%kernels for calculating the spatial derivatives from tps 'sources'
128    end
[581]129    for ilist=1:length(FieldName)
130        switch FieldName{ilist}
[515]131            case 'vec(U,V)'
[1137]132                DataOut.U(ind_sel)=DataOut.U(ind_sel)+weight.*EM *FieldVar(1:nbvec_sub+3,isub,1);
133                DataOut.V(ind_sel)=DataOut.V(ind_sel)+weight.*EM *FieldVar(1:nbvec_sub+3,isub,2);
[515]134            case 'U'
[1137]135                DataOut.U(ind_sel)=DataOut.U(ind_sel)+weight.*EM *FieldVar(1:nbvec_sub+3,isub,1);
[515]136            case 'V'
[1137]137                DataOut.V(ind_sel)=DataOut.V(ind_sel)+weight.*EM *FieldVar(1:nbvec_sub+3,isub,2);
[515]138            case 'norm(U,V)'
[1137]139                U=DataOut.U(ind_sel)+weight.*EM *FieldVar(1:nbvec_sub+3,isub,1);
140                V=DataOut.V(ind_sel)+weight.*EM *FieldVar(1:nbvec_sub+3,isub,2);
[515]141                DataOut.norm(ind_sel)=sqrt(U.*U+V.*V);
142            case 'curl(U,V)'
[1137]143                DataOut.curl(ind_sel)=DataOut.curl(ind_sel)-weight.*EMDY *FieldVar(1:nbvec_sub+3,isub,1)+weight.*EMDX *FieldVar(1:nbvec_sub+3,isub,2);
[515]144            case 'div(U,V)'
[1139]145                DataOut.div(ind_sel)=DataOut.div(ind_sel)+weight.*EMDX*FieldVar(1:nbvec_sub+3,isub,1)+weight.*EMDY*FieldVar(1:nbvec_sub+3,isub,2);
[515]146            case 'strain(U,V)'
[1139]147                DataOut.strain(ind_sel)=DataOut.strain(ind_sel)+weight.*EMDY*FieldVar(1:nbvec_sub+3,isub,1)+weight.*EMDX*FieldVar(1:nbvec_sub+3,isub,2);
[1090]148            case 'DUDX(U,V)'
[1137]149                DataOut.DUDX(ind_sel)=DataOut.DUDX(ind_sel)+weight.*EMDX *FieldVar(1:nbvec_sub+3,isub,1);
[1090]150            case 'DUDY(U,V)'
[1137]151                DataOut.DUDY(ind_sel)=DataOut.DUDY(ind_sel)+weight.*EMDY*FieldVar(1:nbvec_sub+3,isub,1);
[1090]152            case 'DVDX(U,V)'
[1137]153                DataOut.DVDX(ind_sel)=DataOut.DVDX(ind_sel)+weight.*EMDX*FieldVar(1:nbvec_sub+3,isub,2);
[1090]154            case 'DVDY(U,V)'
[1137]155                DataOut.DVDY(ind_sel)=DataOut.DVDY(ind_sel)+weight.*EMDY *FieldVar(1:nbvec_sub+3,isub,2);
[515]156        end
157    end
158end
[867]159nbval(nbval==0)=NaN;% to avoid division by zero for averaging
[515]160ListFieldOut=fieldnames(DataOut);
161for ifield=1:numel(ListFieldOut)
162    DataOut.(ListFieldOut{ifield})=DataOut.(ListFieldOut{ifield})./nbval;
163    DataOut.(ListFieldOut{ifield})=reshape(DataOut.(ListFieldOut{ifield}),npy,npx);
164end
165
166
167
168
Note: See TracBrowser for help on using the repository browser.