OOPS
qg_projection_mod.F90
Go to the documentation of this file.
1 ! (C) Copyright 2009-2016 ECMWF.
2 !
3 ! This software is licensed under the terms of the Apache Licence Version 2.0
4 ! which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
5 ! In applying this licence, ECMWF does not waive the privileges and immunities
6 ! granted to it by virtue of its status as an intergovernmental organisation nor
7 ! does it submit to any jurisdiction.
8 
10 
11 use kinds
13 
14 implicit none
15 
16 private
17 public :: xy_to_lonlat,lonlat_to_xy
18 ! ------------------------------------------------------------------------------
19 contains
20 ! ------------------------------------------------------------------------------
21 !> Convert x/y to lon/lat
22 subroutine xy_to_lonlat(x,y,lon,lat,mapfac)
23 
24 ! Passed variables
25 real(kind_real), intent(in) :: x
26 real(kind_real), intent(in) :: y
27 real(kind_real), intent(out) :: lon
28 real(kind_real), intent(out) :: lat
29 real(kind_real), intent(out), optional :: mapfac
30 
31 ! Local variables
32 real(kind_real) :: rlon,rlat,rlon_c,rlat_c,x_c,y_c
33 
34 ! Test x/y
35 if ((x<0.0).or.(x>domain_zonal)) call abor1_ftn('xy_to_lonlat: x point out of the domain')
36 if ((y<0.0).or.(y>domain_meridional)) call abor1_ftn('xy_to_lonlat: y point out of the domain')
37 
38 ! Define central point
39 rlon_c = 0.0
40 rlat_c = asin(f0/(2.0*omega))
41 x_c = 0.5*domain_zonal
42 y_c = 0.5*domain_meridional
43 
44 ! Define longitude/latitude/map factor
45 rlon = (x-x_c)/req+rlon_c
46 rlat = 2.0*atan2(exp(y/req),exp(y_c/req))-0.5*pi+rlat_c
47 if (present(mapfac)) mapfac = 1.0/cos(rlat)
48 
49 ! Convert lon to radians
50 lon = rlon*rad_to_deg
51 lat = rlat*rad_to_deg
52 
53 end subroutine xy_to_lonlat
54 ! ------------------------------------------------------------------------------
55 !> Convert lon/lat to x/y
56 subroutine lonlat_to_xy(lon,lat,x,y)
57 
58 ! Passed variables
59 real(kind_real), intent(in) :: lon
60 real(kind_real), intent(in) :: lat
61 real(kind_real), intent(out) :: x
62 real(kind_real), intent(out) :: y
63 
64 ! Local variables
65 real(kind_real) :: rlon,rlat,rlon_c,rlat_c,x_c,y_c
66 
67 ! Convert to radians
68 rlon = lon*deg_to_rad
69 rlat = lat*deg_to_rad
70 
71 ! Define central point
72 rlon_c = 0.0
73 rlat_c = asin(f0/(2.0*omega))
74 x_c = 0.5*domain_zonal
75 y_c = 0.5*domain_meridional
76 
77 ! Define x/y
78 x = x_c+(rlon-rlon_c)*req
79 y = y_c+log(tan(0.25*pi+0.5*(rlat-rlat_c)))*req
80 
81 ! Test x/y
82 if ((x<0.0).or.(x>domain_zonal)) call abor1_ftn('lonlat_to_xy: x point out of the domain')
83 if ((y<0.0).or.(y>domain_meridional)) call abor1_ftn('lonlat_to_xy: y point out of the domain')
84 
85 end subroutine lonlat_to_xy
86 ! ------------------------------------------------------------------------------
87 end module qg_projection_mod
real(kind_real), parameter, public omega
Rotation rate of the Earth (rad/s)
real(kind_real), parameter, public rad_to_deg
Radians to degrees.
real(kind_real), parameter, public pi
Pi.
real(kind_real), parameter, public req
Earth radius at equator (m)
real(kind_real), parameter, public f0
Coriolis parameter at the center of the domain.
real(kind_real), parameter, public domain_meridional
Model domain in meridional direction (m)
real(kind_real), parameter, public domain_zonal
Model domain in zonal direction (m)
real(kind_real), parameter, public deg_to_rad
Degrees to radians.
subroutine, public xy_to_lonlat(x, y, lon, lat, mapfac)
Convert x/y to lon/lat.
subroutine, public lonlat_to_xy(lon, lat, x, y)
Convert lon/lat to x/y.