m-thesis-documentation/figures/beacon/field/src/beacon_field.py

402 lines
12 KiB
Python
Raw Normal View History

#!/usr/bin/env python3
__doc__ = \
"""
Generate a figure showing a value combining
the delays between a transmitter and a set of antennas
"""
import matplotlib.pyplot as plt
import numpy as np
from itertools import chain, combinations, product
c_light = 3e8
f_beacon = 50e6
class Antenna:
"""
Simple Antenna class
"""
def __init__(self,x=0,y=0,z=0,t0=0,name=""):
self.x = x
self.y = y
self.z = z
self.t = t0
self.name = name
def __repr__(self):
cls = self.__class__.__name__
return f'{cls}(x={self.x!r},y={self.y!r},z={self.z!r},t0={self.t!r},name={self.name!r})'
def phase_mod(phase, low=np.pi):
"""
Modulo phase such that it falls within the
interval $[-low, 2\pi - low)$.
"""
return (phase + low) % (2*np.pi) - low
def dist(a,b):
return ((a.x-b.x)**2+(a.y-b.y)**2+(a.z-b.z)**2)**0.5
def phase(a,b,f=f_beacon,wrap=False):
t = dist(a,b)/c_light
phase = t*f*2*np.pi
if wrap:
phase = phase_mod(phase)
return phase
def grid_plot(grid, text_dx=(0,0), ax=None, plot_kwargs={}, annot_kwargs={}, scaler=None):
if ax is None:
ax = plt.gca()
if scaler is None:
scaler = 1
2022-10-05 18:59:35 +02:00
if not grid:
return
default_plot_kwargs=dict(color='k', marker='x', markersize=10, linestyle='None')
plot_kwargs = {**default_plot_kwargs, **plot_kwargs}
x = [a.x/scaler for a in grid]
y = [a.y/scaler for a in grid]
l = [a.name for a in grid]
ax.plot(x, y, **plot_kwargs)
if annot_kwargs is not None:
for x_,y_,l_ in zip(x,y,l):
ax.annotate(l_,(x_,y_), xytext=(x_+text_dx[0], y_+text_dx[0]), **annot_kwargs)
def antenna_combinations(ants, ref_ant=None):
if ref_ant is not None: # use only one reference antenna for the baselines
ref = ref_ant
ant_combi = [ (ref, j) for j in ants if j is not ref ]
else: # use all baselines
ant_combi = combinations(ants, 2)
return list(ant_combi)
def calculate_field(field, tx, ant_combi, ref_ant=None, calculate_phase=True):
if plot_phase:
p_ij = [ (phase(tx,i) - phase(tx,j)) for i,j in ant_combi ]
else:
t_ij = [ (dist(tx,i) - dist(tx, j))/c_light for i,j in ant_combi ]
val = []
xx = []
yy = []
xs = field[0]
ys = field[1]
for x in xs:
for y in ys:
tx_t = Antenna(x=x,y=y)
if plot_phase: # phase
dp_ij = np.array([ (phase(tx_t,i) - phase(tx_t,j)) - p_ij[k] for k,(i,j) in enumerate(ant_combi) ])
if True: # phase wrap
dp_ij = phase_mod(dp_ij)
val.append(np.sum(dp_ij**2)**0.5)
else: # time
dt_ij = np.array([ (dist(tx_t,i) - dist(tx_t, j))/c_light - t_ij[k] for k,(i,j) in enumerate(ant_combi) ])
val.append(np.sum(dt_ij**2)**0.5)
xx.append(x)
yy.append(y)
return np.array(xx), np.array(yy), np.array(val)
def plot_field(
tx, ants, xx, yy, val,
ref_ant=None, plot_phase=None, mask=None,
color_label='$\\left( t - \\tau \\right)^2$',
ax=None, bin_type='square', colorbar=True,
grid_kwargs={}, scaler=None,
**scatter_kwargs
):
if ax is None:
ax = plt.gca()
ax.set_xlabel('x [m]')
ax.set_ylabel('y [m]')
max_xx, max_yy = max(xx), max(yy)
if max_xx > 1e3 or max_yy > 1e3:
ax.set_xlabel('x [km]')
ax.set_ylabel('y [km]')
scaler = 1e3
dxx = xx[-1] - xx[0]
dyy = yy[-1] - yy[0]
if scaler is None:
scaler = 1
else:
xx = np.array(xx)/scaler
yy = np.array(yy)/scaler
default_scatter_kwargs = {}
default_scatter_kwargs['cmap'] = 'Spectral_r'
if mask is not None:
val[mask] = np.nan
if plot_phase is None:
pass
elif plot_phase:
val_min = min(val)
default_scatter_kwargs['vmax'] = len(ants)*np.pi + val_min
pass
scatter_kwargs = {**default_scatter_kwargs, **scatter_kwargs}
2022-10-05 18:59:35 +02:00
if tx or ants:
grid_plot_kwargs = dict(marker='X', color='w', alpha=0.8, markeredgecolor='k', markeredgewidth=1)
grid_text_kwargs = dict(
fontsize='large',
color='k',
bbox=dict(boxstyle='Round', alpha=0.5, facecolor='w')
)
if 'plot_kwargs' in grid_kwargs:
grid_plot_kwargs = {**grid_plot_kwargs, **grid_kwargs['plot_kwargs']}
if 'text_kwargs' in grid_kwargs:
grid_text_kwargs = {**grid_text_kwargs, **grid_kwargs['text_kwargs']}
if tx:
grid_plot([tx], text_dx=(dxx *0/scaler, 0), ax=ax, plot_kwargs=grid_plot_kwargs, annot_kwargs=grid_text_kwargs, scaler=scaler)
if len(ants) > 3:
grid_text_kwargs = None
grid_plot(ants, text_dx=(dxx*0/scaler, 0), ax=ax, plot_kwargs=grid_plot_kwargs, annot_kwargs=grid_text_kwargs, scaler=scaler)
title = ''
if len(ants) == 1:
title += "Single Antenna"
elif len(ants) == 2:
title += "Single Baseline"
else:
if len(ants) == 3:
title += "Three Baseline"
else:
if ref_ant is not None:
title += "MultiBaseline"
else:
title += "All Baselines"
if ref_ant is not None:
title += " with Reference antenna={}".format(ref_ant.name)
if plot_phase:
title += "\n"
title += "f=${}$MHz".format(f_beacon/1e6)
ax.set_title(title)
if bin_type == 'hex': # hexbin
sc = ax.hexbin(xx, yy, C=val, **scatter_kwargs)
elif bin_type == 'square': # squarebinning
_, _, _, sc = ax.hist2d(xx, yy, weights=val, bins=int(len(xx)**0.5), **scatter_kwargs)
else: # overlayed markers
sc = ax.scatter(xx,yy,c=val, **scatter_kwargs)
if colorbar:
fig = ax.get_figure()
fig.colorbar(sc, ax=ax, label=color_label)
return ax
def square_grid(dx=1, N_x=10, dy=None, N_y=None, x_start=0, y_start=0):
N_y = N_x if N_y is None else N_y
dy = dx if dy is None else dy
return product(
(x_start + n*dx for n in range(N_x)),
(y_start + n*dy for n in range(N_y)),
)
def triangular_grid(dx=1, N_x=10, dy=None, N_y=None, x_start=0, y_start=0):
return chain(
square_grid(dx=dx,dy=dy,N_x=N_x,N_y=N_y,x_start=x_start,y_start=y_start),
square_grid(dx=dx,dy=dy,N_x=N_x,N_y=N_y,x_start=x_start + dx/2,y_start=y_start + dy/2),
)
if __name__ == "__main__":
km = 1e3#m
from argparse import ArgumentParser
import os.path as path
parser = ArgumentParser(description=__doc__)
parser.add_argument("fname", metavar="path/to/figure[/]", nargs="?", help="Location for generated figure, will append __file__ if a directory. If not supplied, figure is shown.")
parser.add_argument("type", choices=['single-left', 'single-center','three-left', 'three-center', 'square', 'tri', 'preset'])
command_group = parser.add_mutually_exclusive_group(required=True)
command_group.add_argument('--time', help='Use the time difference for the field', action='store_false')
command_group.add_argument('--phase', help='Use wrapped phase for the field', action='store_true')
2022-10-05 19:05:19 +02:00
parser.add_argument('--ref', dest='ref_ant', metavar='ref_antenna', type=int, help='Number of antenna to use as reference')
parser.add_argument('--max-rms', dest='max_rms', metavar='max_rms', type=float, help='Maximum rms to show in colorbar', default=True)
parser.add_argument('--zoom', choices=['none', 'tx'], help='Zoom to object', default='none')
args = parser.parse_args()
2023-08-15 16:28:27 +02:00
if False: #specific formatting
from matplotlib import rcParams
#rcParams["text.usetex"] = True
rcParams["font.family"] = "serif"
rcParams["font.size"] = "12"
if True:# small
figsize = (6, 4)
rcParams["font.size"] = "14" # 15 at 6,4 looks fine
elif True: # large
figsize = (9, 6)
rcParams["grid.linestyle"] = 'dotted'
rcParams["figure.figsize"] = figsize
2022-10-18 07:31:51 +02:00
if args.fname == 'none':
args.fname = None
if args.fname is not None:
if path.isdir(args.fname):
args.fname = path.join(args.fname, path.splitext(path.basename(__file__))[0]) # leave off extension
if not path.splitext(args.fname)[1]:
args.fname = [ args.fname+ext for ext in ['.pdf', '.png'] ]
args.plot_phase = args.phase or args.time
del args.time, args.phase
if 'single' in args.type or 'three' in args.type: # single baseline
### Field
x_low, x_high, N_x = -5*km, 5*km, 151
y_low, y_high, N_y = -5*km, 5*km, 151
### Geometry
ants = [
Antenna(x=-.75*km,y=0,z=0,name="a"),
Antenna(x=.75*km, y=0,z=0,name="b"),
]
if 'three' in args.type:
ants.append(Antenna(x=0, y=-.75*km*np.sqrt(2),z=0, name='c'))
if 'center' in args.type:
tx = Antenna(x=-000,y=4*km,z=0,name="tx")
else:
tx = Antenna(x=-4*km,y=4*km,z=0,name="tx")
elif args.type == 'square' or args.type == 'tri': # from grid definition
### Field
x_low, x_high, N_x = -1800, 1800, 261
y_low, y_high, N_y = -x_low, -x_high, N_x
### Geometry
tx = Antenna(x=-800,y=300,z=0,name="tx")
x_start, dx, ant_N_x = 0, 50, 3
y_start, dy, ant_N_y = 0, dx, ant_N_x
if args.type == 'square': # square grid
grid_func = square_grid
elif args.type == 'tri': # triangular
grid_func = triangular_grid
grid = grid_func(dx=dx, dy=dy, N_x=ant_N_x, N_y=ant_N_y, x_start=x_start, y_start=y_start)
ants = [ Antenna(x=x,y=y,z=0,name=i) for i, (x,y) in enumerate(grid) ]
else:
### Field
x_low, x_high, N_x = -400, 400, 161
y_low, y_high, N_y = -x_low, -x_high, N_x
### Geometry
tx = Antenna(x=-300,y=300,z=0,name="tx")
ants = [
Antenna(x=100,y=0,z=0,name="a"),
Antenna(x=0,y=-50,z=0,name="b"),
Antenna(x=+50,y=-180,z=0,name="c"),
Antenna(x=125,y=180,z=0,name="d"),
]
if args.zoom != 'none': # tx zoom
if args.zoom == 'tx':
x_low, x_high, N_x = tx.x - 250, tx.x + 250, 151
y_low, y_high, N_y = x_low, x_high, N_x
else:
raise NotImplementedError# args.zoom
###
### Options
###
plot_phase = args.plot_phase
if args.ref_ant is not None:
2022-10-05 19:05:19 +02:00
ref_ant = ants[args.ref_ant]
else:
ref_ant = None
ant_combi = antenna_combinations(ants, ref_ant=ref_ant)
xs = np.linspace(x_low, x_high, N_x)
ys = np.linspace(y_low, y_high, N_y)
xx, yy, val = calculate_field((xs, ys), tx, ant_combi, calculate_phase=plot_phase)
kwargs = {}
mask = None
if plot_phase:
2023-08-15 16:28:27 +02:00
color_label='$\\sqrt{ \\sum_{(i,j)} \\left(\\Delta\\varphi_{ij}(x) - \\Delta \\varphi_{ij}\\right)^2}$ [rad]'
if args.max_rms:
min_val = abs(min(min(val), args.max_rms))
mask = abs(val) > np.pi + min_val
else:
color_label='$\\sqrt{ \\sum_{(i,j)} \\left(\Delta t_{ij}(x) - \\Delta t_{ij}\\right)^2}$ [ns]'
val *= 1e9
if args.max_rms:
kwargs['vmax'] = 100 if args.max_rms is True else args.max_rms
if not True:
mask = abs(val) > 1.1*kwargs['vmax']
if mask is not None:
ax = plot_field([], [], xx, yy, val, cmap='Greys', colorbar=False, alpha=0.5)
ax = plot_field(tx, ants, xx, yy, val, ref_ant=ref_ant, mask=mask, color_label=color_label, **kwargs)
# if plot_phase:
# N_lowest = np.min(len(ant_combi)-1, 10)
# lowest_idx = np.argpartition(val, N_lowest)[:N_lowest]
#
# print(lowest_idx)
# print(val[lowest_idx])
# print( list(zip(np.array(xx)[lowest_idx], np.array(yy)[lowest_idx])) )
if args.fname is not None:
if isinstance(args.fname, str):
args.fname = [args.fname]
2023-08-15 16:28:27 +02:00
fig = plt.gcf()
fig.tight_layout()
for fname in args.fname:
plt.savefig(fname)
else:
plt.show()