diff --git a/misc/finiterectlat-constant-driving.py b/misc/finiterectlat-constant-driving.py index a46861a..06c70be 100755 --- a/misc/finiterectlat-constant-driving.py +++ b/misc/finiterectlat-constant-driving.py @@ -4,7 +4,7 @@ import math from qpms.argproc import ArgParser, make_dict_action, sslice, annotate_pdf_metadata figscale=3 -ap = ArgParser(['rectlattice2d_finite', 'single_particle', 'single_lMax', 'single_omega']) +ap = ArgParser(['rectlattice2d_finite', 'single_particle', 'single_lMax_or_vswfset', 'single_omega']) ap.add_argument("-k", '--wavevector', nargs=2, type=float, required=True, help='"Bloch" vector, modulating phase of the driving', metavar=('KX', 'KY'), default=(0., 0.)) # ap.add_argument("--kpi", action='store_true', help="Indicates that the k vector is given in natural units instead of SI, i.e. the arguments given by -k shall be automatically multiplied by pi / period (given by -p argument)") ap.add_argument("-o", "--output", type=str, required=False, help='output path (if not provided, will be generated automatically)') @@ -36,8 +36,8 @@ px, py = a.period particlestr = ("sph" if a.height is None else "cyl") + ("_r%gnm" % (a.radius*1e9)) if a.height is not None: particlestr += "_h%gnm" % (a.height * 1e9) -defaultprefix = "cd_%s_p%gnmx%gnm_%dx%d_m%s_n%s_k_%g_%g_f%geV_L%d_micro-%s" % ( - particlestr, px*1e9, py*1e9, Nx, Ny, str(a.material), str(a.background), a.wavevector[0], a.wavevector[1], a.eV, a.lMax, "SO3" if a.symmetry_adapted is None else a.symmetry_adapted) +defaultprefix = "cd_%s_p%gnmx%gnm_%dx%d_m%s_n%s_k_%g_%g_f%geV_%s_micro-%s" % ( + particlestr, px*1e9, py*1e9, Nx, Ny, str(a.material), str(a.background), a.wavevector[0], a.wavevector[1], a.eV, ap.bspecstr, "SO3" if a.symmetry_adapted is None else a.symmetry_adapted) logging.info("Default file prefix: %s" % defaultprefix) import numpy as np @@ -126,7 +126,7 @@ orig_xy = np.stack(np.meshgrid(orig_x, orig_y), axis = -1) omega = ap.omega -bspec = BaseSpec(lMax = a.lMax) +bspec = ap.bspec medium = EpsMuGenerator(ap.background_epsmu) particles= [Particle(orig_xy[i], ap.tmgen, bspec) for i in np.ndindex(orig_xy.shape[:-1])] diff --git a/misc/finiterectlat-modes.py b/misc/finiterectlat-modes.py index 701d61d..1d6461f 100755 --- a/misc/finiterectlat-modes.py +++ b/misc/finiterectlat-modes.py @@ -4,7 +4,7 @@ import math from qpms.argproc import ArgParser, annotate_pdf_metadata -ap = ArgParser(['rectlattice2d_finite', 'background_analytical', 'single_particle', 'single_lMax', ]) +ap = ArgParser(['rectlattice2d_finite', 'background_analytical', 'single_particle', 'single_lMax_or_vswfset', ]) ap.add_argument("-t", "--rank-tolerance", type=float, default=1e11) ap.add_argument("-c", "--min-candidates", type=int, default=1, help='always try at least this many eigenvalue candidates, even if their SVs in the rank tests are lower than rank_tolerance') @@ -35,8 +35,8 @@ thegroup = 'D4h' if px == py and Nx == Ny and not a.D2 else 'D2h' particlestr = ("sph" if a.height is None else "cyl") + ("_r%gnm" % (a.radius*1e9)) if a.height is not None: particlestr += "_h%gnm" % (a.height * 1e9) -defaultprefix = "%s_p%gnmx%gnm_%dx%d_m%s_B%s_L%d_c(%s±%g±%gj)eV_cn%d_%s" % ( - particlestr, px*1e9, py*1e9, Nx, Ny, str(a.material), str(a.background), a.lMax, +defaultprefix = "%s_p%gnmx%gnm_%dx%d_m%s_B%s_%s_c(%s±%g±%gj)eV_cn%d_%s" % ( + particlestr, px*1e9, py*1e9, Nx, Ny, str(a.material), str(a.background), ap.bspecstr, str(a.centre), a.ar, a.ai, a.N, thegroup, ) @@ -70,7 +70,7 @@ orig_y = (np.arange(Ny/2) + (0 if (Ny % 2) else .5)) * py orig_xy = np.stack(np.meshgrid(orig_x, orig_y), axis = -1) -bspec = BaseSpec(lMax = a.lMax) +bspec = ap.bspec medium = EpsMuGenerator(ap.background_epsmu) particles= [Particle(orig_xy[i], ap.tmgen, bspec) for i in np.ndindex(orig_xy.shape[:-1])] diff --git a/misc/finiterectlat-scatter.py b/misc/finiterectlat-scatter.py index 82eae39..2e8c481 100755 --- a/misc/finiterectlat-scatter.py +++ b/misc/finiterectlat-scatter.py @@ -5,7 +5,7 @@ pi = math.pi from qpms.argproc import ArgParser -ap = ArgParser(['rectlattice2d_finite', 'single_particle', 'single_lMax', 'omega_seq_real_ng', 'planewave']) +ap = ArgParser(['rectlattice2d_finite', 'single_particle', 'single_lMax_or_vswfset', 'omega_seq_real_ng', 'planewave']) ap.add_argument("-o", "--output", type=str, required=False, help='output path (if not provided, will be generated automatically)') ap.add_argument("-O", "--plot-out", type=str, required=False, help="path to plot output (optional)") ap.add_argument("-P", "--plot", action='store_true', help="if -p not given, plot to a default path") @@ -36,8 +36,8 @@ px, py = a.period particlestr = ("sph" if a.height is None else "cyl") + ("_r%gnm" % (a.radius*1e9)) if a.height is not None: particlestr += "_h%gnm" % (a.height * 1e9) -defaultprefix = "%s_p%gnmx%gnm_%dx%d_m%s_bg%s_φ%gπ_θ(%g_%g)π_ψ%gπ_χ%gπ_f%s_L%d" % ( - particlestr, px*1e9, py*1e9, Nx, Ny, str(a.material), str(a.background), a.phi/pi, np.amin(a.theta)/pi, np.amax(a.theta)/pi, a.psi/pi, a.chi/pi, ap.omega_descr, a.lMax, ) +defaultprefix = "%s_p%gnmx%gnm_%dx%d_m%s_bg%s_φ%gπ_θ(%g_%g)π_ψ%gπ_χ%gπ_f%s_%s" % ( + particlestr, px*1e9, py*1e9, Nx, Ny, str(a.material), str(a.background), a.phi/pi, np.amin(a.theta)/pi, np.amax(a.theta)/pi, a.psi/pi, a.chi/pi, ap.omega_descr, ap.bspecstr, ) logging.info("Default file prefix: %s" % defaultprefix) @@ -47,7 +47,7 @@ orig_y = (np.arange(Ny/2) + (0 if (Ny % 2) else .5)) * py orig_xy = np.stack(np.meshgrid(orig_x, orig_y), axis = -1) -bspec = BaseSpec(lMax = a.lMax) +bspec = ap.bspec particles= [Particle(orig_xy[i], ap.tmgen, bspec=bspec) for i in np.ndindex(orig_xy.shape[:-1])] sym = FinitePointGroup(point_group_info['D2h']) diff --git a/misc/rectlat_simple_modes.py b/misc/rectlat_simple_modes.py index b39b5e4..13b3083 100755 --- a/misc/rectlat_simple_modes.py +++ b/misc/rectlat_simple_modes.py @@ -3,7 +3,7 @@ import math from qpms.argproc import ArgParser, annotate_pdf_metadata -ap = ArgParser(['rectlattice2d', 'const_real_background', 'single_particle', 'single_lMax']) # const_real_background needed for calculation of the diffracted orders +ap = ArgParser(['rectlattice2d', 'const_real_background', 'single_particle', 'single_lMax_or_vswfset']) # const_real_background needed for calculation of the diffracted orders ap.add_argument("-k", nargs=2, type=float, required=True, help='k vector', metavar=('K_X', 'K_Y')) ap.add_argument("--kpi", action='store_true', help="Indicates that the k vector is given in natural units instead of SI, i.e. the arguments given by -k shall be automatically multiplied by pi / period (given by -p argument)") ap.add_argument("--rank-tol", type=float, required=False) @@ -28,8 +28,8 @@ if a.kpi: particlestr = ("sph" if a.height is None else "cyl") + ("_r%gnm" % (a.radius*1e9)) if a.height is not None: particlestr += "_h%gnm" % (a.height * 1e9) -defaultprefix = "%s_p%gnmx%gnm_m%s_n%g_b%+d_k(%g_%g)um-1_L%d_cn%d" % ( - particlestr, px*1e9, py*1e9, str(a.material), a.refractive_index, a.band_index, a.k[0]*1e-6, a.k[1]*1e-6, a.lMax, a.N) +defaultprefix = "%s_p%gnmx%gnm_m%s_n%g_b%+d_k(%g_%g)um-1_%s_cn%d" % ( + particlestr, px*1e9, py*1e9, str(a.material), a.refractive_index, a.band_index, a.k[0]*1e-6, a.k[1]*1e-6, ap.bspecstr, a.N) import logging logging.basicConfig(format='%(asctime)s %(message)s', level=logging.INFO) @@ -91,7 +91,7 @@ freqradius = .5 * (top - bottom) * a.interval_factor centfreq = bottom + freqradius if a.band_index > 0 else top - freqradius -bspec = BaseSpec(lMax = a.lMax) +bspec = ap.bspec pp = Particle(orig_xy[0][0], t = ap.tmgen, bspec=bspec) ss, ssw = ScatteringSystem.create([pp], ap.background_emg, centfreq, latticebasis = ap.direct_basis) diff --git a/qpms/argproc.py b/qpms/argproc.py index bdbffb2..afc93a8 100644 --- a/qpms/argproc.py +++ b/qpms/argproc.py @@ -323,6 +323,12 @@ class ArgParser: action=make_dict_action(argtype=hashable_complex_matrix, postaction='store', first_is_key=True), help='constant T-matrix (elements must correspond to ++vswf-set)') + def __add_single_lMax_or_vswfset_group(ap): + grp = ap.add_mutually_exclusive_group(required=True) + grp.add_argument("-w", "--vswf-set", type=string2bspec_tuple, + help='Manual specification of VSWF set codes (format as a python list of integers); see docs on qpms_uvswfi_t for valid codes or simply use --lMax instead. Overrides --lMax.') + grp.add_argument("-L", "--lMax", type=int, default=3, help='multipole degree cutoff') + atomic_arguments = { 'rectlattice2d_periods': lambda ap: ap.add_argument("-p", "--period", type=float, nargs='+', required=True, help='square/rectangular lattice periods', metavar=('px','[py]')), 'rectlattice2d_counts': lambda ap: ap.add_argument("--size", type=int, nargs=2, required=True, help='rectangular array size (particle column, row count)', metavar=('NCOLS', 'NROWS')), @@ -338,6 +344,7 @@ class ArgParser: 'bg_real_refractive_index': lambda ap: ap.add_argument("-n", "--refractive-index", type=float, default=1., help='background medium strictly real refractive index'), 'bg_analytical': lambda ap: ap.add_argument("-B", "--background", type=material_spec, default=1., help="Background medium specification (constant real or complex refractive index, or supported material label)"), 'single_lMax': lambda ap: ap.add_argument("-L", "--lMax", type=int, required=True, default=3, help='multipole degree cutoff'), + 'single_lMax_or_vswfset': __add_single_lMax_or_vswfset_group, 'single_lMax_extend': lambda ap: ap.add_argument("--lMax-extend", type=int, required=False, default=6, help='multipole degree cutoff for T-matrix calculation (cylindrical particles only'), 'outfile': lambda ap: ap.add_argument("-o", "--output", type=str, required=False, help='output path (if not provided, will be generated automatically)'), # TODO consider type=argparse.FileType('w') 'plot_out': lambda ap: ap.add_argument("-O", "--plot-out", type=str, required=False, help="path to plot output (optional)"), @@ -354,6 +361,7 @@ class ArgParser: 'single_particle': ("Single particle definition (shape [currently spherical or cylindrical]) and materials, incl. background)", ('background',), ('single_material', 'single_radius', 'single_height', 'single_lMax_extend'), ('_eval_single_tmgen',), ()), 'multi_particle': ("One or more particle definition (shape [curently spherical or cylindrical]), materials, and positions)", ('background',), ('multi_particle',), ('_process_multi_particle',), ()), 'single_lMax': ("Single particle lMax definition", (), ('single_lMax',), (), ()), + 'single_lMax_or_vswfset': ("Single particle lMax definition", (), ('single_lMax_or_vswfset',), ('_eval_single_lMax_or_vswfset',), ()), 'single_omega': ("Single angular frequency", (), ('single_frequency_eV',), ('_eval_single_omega',), ()), 'omega_seq': ("Equidistant real frequency range with possibility of adding individual frequencies", (), ('seq_frequency_eV', 'multiple_frequency_eV_optional',), ('_eval_omega_seq',), ()), 'omega_seq_real_ng': ("Equidistant real frequency ranges or individual frequencies (new syntax)", (), ('real_frequencies_eV_ng',), ('_eval_omega_seq_real_ng',), ()), @@ -501,6 +509,16 @@ class ArgParser: from .constants import eV, hbar self.omega = self.args.eV * eV / hbar + def _eval_single_lMax_or_vswfset(self): #feature: single_lMax_or_vswfset + from .cybspec import BaseSpec + a = self.args + if a.vswf_set is not None: + self.bspec = BaseSpec(a.vswf_set) + self.bspecstr = "WF%s" % (str(a.vswf_set),) + else: + self.bspec = BaseSpec(lMax=a.lMax) + self.bspecstr = "L%d" % (a.lMax,) + def _eval_omega_seq(self): # feature: omega_seq import numpy as np from .constants import eV, hbar