3.3. Tensor formulation of photoionization#
A number of authors have treated PADs and related problems in the context of photoionization theory and matrix element reconstruction (see Quantum Metrology Vol. 2 [9] Chpt. 8 for examples and discussion, a range of review articles can also be found in the literature, e.g. Refs. [91, 92, 93, 94]); herein, a geometric tensor based formalism is developed, which is close in spirit to the treatments given by Underwood and co-workers [88, 95, 96], but further separates various sets of physical parameters into dedicated tensors; this allows for a unified theoretical and numerical treatment, where the latter computes properties as tensor variables which can be further manipulated and investigated to give detailed insights into various aspects of photoionization for the system at hand, and implications/effects for matrix element retrieval in a given case. Furthermore, the tensors can readily be converted to a density matrix representation [83, 97], which is more natural for some quantities, and also emphasizes the link to quantum state tomography and other quantum information techniques. Much of the theoretical background, as well as application to aspects of the current problem, can be found in the textbooks of Blum [97] and Zare [83].
Within this treatment, the observables can be defined in a series of simplified forms, emphasizing the quantities of interest for a given problem. The most general, and simplest, form is given in Sect. 3.3.1, in terms of channel functions, and the remainder of this section (Sect. 3.3.2 - Section 3.3.9) gives a detailed breakdown of the various components of the channel functions, and numerical examples.
3.3.1. Channel functions#
A simple form of the equations[1], amenable to fitting and numerical implementation, is to write the observables in terms of channel functions, which define the ionization continuum for a given case and set of parameters
Where
Eq. (3.13) is effectively a convolution equation (cf. Refs. [95, 98]) with channel functions
These complex matrix elements can also be equivalently defined in a magnitude, phase form:
This tensorial form is numerically implemented in the ePSproc [34] codebase, and is in contradistinction to standard numerical routines in which the requisite terms are usually computed from vectorial and/or nested summations. The Photoelectron Metrology Toolkit [5] codebase implements radial matrix elements retrieval based on the tensor formalism, with pre-computation of all the geometric tensor components (channel functions) prior to a fitting protocol for matrix element analysis, essentially a fit to Eqn. (3.13), with terms
3.3.2. Full tensor expansion#
In more detail, the channel functions
For the MF:
In both cases a set of geometric tensor terms are required, these terms provide details of:
: polarization geometry & coupling with the electric field. : geometric coupling of the partial waves into the terms (spherical tensors). Note for the AF case the terms may be reindexed by , which allows for the projection dependence on the ADMs (see below). , : frame couplings and rotations (note slightly different terms for MF and AF). : ensemble alignment described as a set of axis distribution moments (ADMs, LF/AF only). Note for a one-photon ionization case - the traditional LF experiment - there will only be a single term, , with no time-dependence, which describes an isotropic molecular ensemble. In general only the AF is discussed explicitly herein, but it is of note that this is identical to the traditional LF definition for this limiting case of an isotropic ensemble.Square-brackets are short-hand for degeneracy terms, e.g.
.
Finally,
Following the tensor components detailed above, the full form of the channel functions of Eq. (3.13) for the AF and MF can be written as:
Note that, in this case as given, time-dependence arises purely from the
It should be emphasized, however, that the underlying physical quantities are essentially identical in all the theoretical approaches, with a set of coupled angular-momenta defining the geometric coupling parameters part of the photoionization problem, despite these differences in the details of the theory and notation.
The various tensors defined above are implemented as functions in the ePSproc codebase [33, 34, 35], and further wrapped for fitting cases in the Photoelectron Metrology Toolkit [5]. In the remainder of this section, numerical examples using these codes are illustrated and explored. Full computational details can be found in the ePSproc documentation [35], including extended discussion of each tensor and complete function references in the geomCalc submodule documentation.
3.3.3. Frame definitions#
A conceptual overview of the LF/AF and relation to the MF, in the context of the bootstrap reconstruction protocol, can be found in Fig. 1.1. A more detailed definition is given in Fig. 3.1, as pertains to the use of angular momentum notation and projections. The figure shows the general use of angular momenta and associated projection terms in molecular spectroscopy as an aid to visualising the discussion in the following sections. Note, however, that some alternative notations are used in this volume, in particular specific projection terms may be used for certain physical quantities.
In simple cases, the frame definition for the AF is identical to that of the LF, since it is usually defined by the laser polarization, with the distinction that an aligned molecular ensemble is additionally present. For the limiting case of an isotropic distribution, the AF and (traditional) LF are identical. However, in cases with non-linear laser polarization, and/or multiple pulses with different polarization vectors, the situation may be more complicated, and additional frame rotation(s) may be required (see Fig. 3.1 inset). In such cases the reference frame may be chosen as the final ionizing laser pulse polarization, or as a symmetry axis in the AF. For high degrees of (3D) alignment the AF may approach the MF in the ideal case, although will usually be limited by the symmetry of the system.

Fig. 3.1 Reference frame and angular momentum definitions for the Laboratory frame (LF) and Molecular frame (MF), using a general notation from molecular spectroscopy. In this case the LF shows an angular momentum vectors
3.3.4. Numerical aside: symmetry-defined channel functions#
In the following sub-sections, each component is defined in detail, including numerical examples. For illustration purposes, the numerical example uses a minimal set of assumptions, and is defined initially purely by symmetry, although further terms may be required for computation of some of the geometric terms and are discussed where required. A fuller discussion of symmetry considerations in photoionization can be found in Sect. 3.2, and discussion of symmetrized harmonics in Sect. 3.6.2.
For this example, the
Show code cell content
# Run default config - may need to set full path here
%run '../scripts/setup_notebook.py'
# Override plotters backend?
# plotBackend = 'pl'
# Set Matplotlib inline for lmPlot() with Seaborn display
%matplotlib inline
*** Setting up notebook with standard Quantum Metrology Vol. 3 imports...
For more details see https://pemtk.readthedocs.io/en/latest/fitting/PEMtk_fitting_basic_demo_030621-full.html
To use local source code, pass the parent path to this script at run time, e.g. "setup_fit_demo ~/github"
*** Running: 2023-12-07 10:45:06
Working dir: /home/jovyan/jake-home/buildTmp/_latest_build/html/doc-source/part1
Build env: html
None
* Loading packages...
* sparse not found, sparse matrix forms not available.
* natsort not found, some sorting functions not available.
* Setting plotter defaults with epsproc.basicPlotters.setPlotters(). Run directly to modify, or change options in local env.
* Set Holoviews with bokeh.
* pyevtk not found, VTK export not available.
* Set Holoviews with bokeh.
Jupyter Book : 0.15.1
External ToC : 0.3.1
MyST-Parser : 0.18.1
MyST-NB : 0.17.2
Sphinx Book Theme : 1.0.1
Jupyter-Cache : 0.6.1
NbClient : 0.7.4
#*** Setup symmetry-defined matrix elements using PEMtk
# Import class
from pemtk.sym.symHarm import symHarm
#*** Compute hamronics for Td, lmax=4
sym = 'D2h'
lmax=4
lmaxPlot = 2 # Set lmaxPlot for subselection on plots later.
# Create symHarm object with given settings,
# this will also compute the symmetrized harmonics
symObj = symHarm(sym,lmax)
Show code cell output
*** Mapping coeffs to ePSproc dataType = matE
Remapped dims: {'C': 'Cont', 'mu': 'it'}
Added dim Eke
Added dim Targ
Added dim Total
Added dim mu
Added dim Type
Found dipole symmetries:
{'B1u': {'m': [0], 'pol': ['z']}, 'B2u': {'m': [-1, 1], 'pol': ['y']}, 'B3u': {'m': [-1, 1], 'pol': ['x']}}
Show code cell content
# Glue items for later
glue("symHarmPGmatE", sym, display=False)
glue("symHarmLmaxmatE", lmax, display=False)
glue("symHarmBasislmaxPlot", lmaxPlot, display=False)
# Display results (real harmonics)
symObj.displayXlm(setCols='h') #, dropLevels='mu')
# Glue version for JupyterBook output
# As above, but with PD object return and glue.
glue("D2hXlm",symObj.displayXlm(setCols='h',returnPD=True), display=False)
Show code cell output
b | |||||||||
---|---|---|---|---|---|---|---|---|---|
h | 0 | 1 | 2 | 3 | 4 | 5 | |||
Character ($\Gamma$) | PFIX ($\mu$) | l | m | ||||||
A1g | 0 | 0 | 0 | 1.0 | |||||
2 | 0 | 1.0 | |||||||
2 | 1.0 | ||||||||
4 | 0 | 1.0 | |||||||
2 | 1.0 | ||||||||
4 | 1.0 | ||||||||
A1u | 0 | 3 | -2 | 1.0 | |||||
B1g | 0 | 2 | -2 | 1.0 | |||||
4 | -4 | 1.0 | |||||||
-2 | 1.0 | ||||||||
B1u | 0 | 1 | 0 | 1.0 | |||||
3 | 0 | 1.0 | |||||||
2 | 1.0 | ||||||||
B2g | 0 | 2 | 1 | 1.0 | |||||
4 | 1 | 1.0 | |||||||
3 | 1.0 | ||||||||
B2u | 0 | 1 | -1 | 1.0 | |||||
3 | -3 | 1.0 | |||||||
-1 | 1.0 | ||||||||
B3g | 0 | 2 | -1 | 1.0 | |||||
4 | -3 | 1.0 | |||||||
-1 | 1.0 | ||||||||
B3u | 0 | 1 | 1 | 1.0 | |||||
3 | 1 | 1.0 | |||||||
3 | 1.0 |
b | |||||||||
---|---|---|---|---|---|---|---|---|---|
h | 0 | 1 | 2 | 3 | 4 | 5 | |||
Character ($\Gamma$) | PFIX ($\mu$) | l | m | ||||||
A1g | 0 | 0 | 0 | 1.0 | |||||
2 | 0 | 1.0 | |||||||
2 | 1.0 | ||||||||
4 | 0 | 1.0 | |||||||
2 | 1.0 | ||||||||
4 | 1.0 | ||||||||
A1u | 0 | 3 | -2 | 1.0 | |||||
B1g | 0 | 2 | -2 | 1.0 | |||||
4 | -4 | 1.0 | |||||||
-2 | 1.0 | ||||||||
B1u | 0 | 1 | 0 | 1.0 | |||||
3 | 0 | 1.0 | |||||||
2 | 1.0 | ||||||||
B2g | 0 | 2 | 1 | 1.0 | |||||
4 | 1 | 1.0 | |||||||
3 | 1.0 | ||||||||
B2u | 0 | 1 | -1 | 1.0 | |||||
3 | -3 | 1.0 | |||||||
-1 | 1.0 | ||||||||
B3g | 0 | 2 | -1 | 1.0 | |||||
4 | -3 | 1.0 | |||||||
-1 | 1.0 | ||||||||
B3u | 0 | 1 | 1 | 1.0 | |||||
3 | 1 | 1.0 | |||||||
3 | 1.0 |
Fig. 3.2 Symmetrized harmonics coefficients (libmsym
[64, 65]. Note that, in this case, the coeffcients have the property that
To compute basis tensors from these symmetry coefficients, they can be converted to the standard radial matrix elements format used in the Photoelectron Metrology Toolkit [5] (see Sect. 6.3.4 for further discussion of the numerical implementation), and used with the standard routines. These are demonstrated below, for two flavours - the base ePSproc [34] routine for computation of AF-PADs, and the Photoelectron Metrology Toolkit [5] fitting routine which wraps this functionality. Note that all tensors are stored as Xarray
[53, 54] objects, which allow for easy numerical manipulation, subselection etc.
#*** Compute basis functions for given matrix elements using PEMtk fit class
# This illustration uses the symmetrized matrix elements set above
#*** To use ePSproc/PEMtk classes,
# these values can be converted to ePSproc BLM data type...
# Run conversion - the default is to set the coeffs to the 'BLM' data type,
# additional dim mappings can also be set.
# Outputs are set to symObj.coeffs[dataType]
dimMap = {'C':'Cont','mu':'muX'}
symObj.toePSproc(dimMap=dimMap)
# Run conversion with a different dimMap & dataType
# Outputs are set to symObj.coeffs[dataType]
dataType = 'matE'
symObj.toePSproc(dimMap = dimMap, dataType=dataType)
#*** Setup class object
data = pemtkFit()
# Set to new key in data class
dataKey = sym
data.data[dataKey] = {}
# Set data.data[dataKey][dataType] from cases set above
# This pushes the symmetrized coeffs computed above to the PEMtk fit class
# object for general use with PEMtk methods.
# Here set 'matE' for use as matrix elements, and 'BLM' for pad plotting routines.
for dataType in ['matE','BLM']:
# Select expansion in complex harmonics, and sum redundant dims
data.data[dataKey][dataType] = symObj.coeffs[dataType]['b (comp)'].sum(['h','muX'])
# Propagate attrs
data.data[dataKey][dataType].attrs = symObj.coeffs[dataType].attrs
# Set data by key
# data.subKey is the default location used by the PEMtk routines
data.subKey = dataKey
#*** Compute basis function - two flavours
# Using PEMtk `afblmMatEfit` method
# - this only returns the product basis set as used for fitting
# See the docs for more details, https://pemtk.readthedocs.io
phaseConvention='S' # For consistency in the method, explicitly set the
# phase convention used here ('S' = standard, 'E' = ePS).
BetaNormX, basisProduct = data.afblmMatEfit(selDims={},
sqThres=False,
phaseConvention=phaseConvention)
# Using ePSproc directly - this includes full basis return if specified
# See the docs for more details, https://epsproc.readthedocs.io
BetaNormX2, basisFull = ep.geomFunc.afblmXprod(data.data[data.subKey]['matE'],
basisReturn = 'Full',
thres=None, selDims={},
sqThres=False,
phaseConvention=phaseConvention)
# The basis dictionary contains various numerical parameters, these are investigated below.
# See also the ePSproc docs at
# https://epsproc.readthedocs.io/en/latest/methods/geometric_method_dev_260220_090420_tidy.html
print(f"Product basis elements: {basisProduct.keys()}")
print(f"Full basis elements: {basisFull.keys()}")
# Use full basis for following sections
basis = basisFull
*** Mapping coeffs to ePSproc dataType = BLM
Remapped dims: {'C': 'Cont', 'mu': 'muX'}
Added dim Eke
Added dim P
Added dim T
Added dim C
*** Mapping coeffs to ePSproc dataType = matE
Remapped dims: {'C': 'Cont', 'mu': 'muX'}
Added dim Eke
Added dim Targ
Added dim Total
Added dim mu
Added dim it
Added dim Type
Product basis elements: dict_keys(['BLMtableResort', 'polProd', 'phaseConvention', 'BLMRenorm'])
Full basis elements: dict_keys(['QNs', 'EPRX', 'lambdaTerm', 'BLMtable', 'BLMtableResort', 'AFterm', 'AKQS', 'polProd', 'phaseConvention', 'BLMRenorm', 'matEmult'])
3.3.5. Matrix element geometric coupling term #
The coupling of the partial-waves as coherent pairs,
Note that this term is equivalent, effectively, to a triple integral over spherical harmonics (e.g. Eq. 3.119 in Zare [83]):
And a similar term appears in the contraction over a pair of harmonics into a resultant harmonic (e.g. Eqs. C.21, C.22 in Blum [97]) - this is how the term arises in the derivation of the observables.
Note also some definitions use conjugate spherical harmonics, which can be converted as, e.g., Eq. C.21 in Blum [97]:
In the current Photoelectron Metrology Toolkit [5] codebase, the relevant basis item can be inspected as below, in order to illustrate the sensitivity of different
The code cells below illustrate this for the current example case, and Fig. 3.3 offers a general summary. In general, this is a convenient way to visualize the selection rules into the observable: for instance, only terms
Numerically, various standard functions may be used to quickly gain deeper insight, for example min/max, averages etc. Such considerations may provide a quick sanity-check for a given case, and may prove useful when planning experiments to investigate particular aspects or channels of a given system. Other properties of the basis functions may also be interrogated numerically; for instance, correlation maps provide an alternative way to check which terms are strongly correlated or coupled, or will dominate a given aspect of the observable.
#*** Tabulate basis
basisKey = 'BLMtableResort' # Key for BLM basis set
# Reformat basis for display (optional)
stackDims = {'LM':['L','M']}
basisPlot = basis[basisKey].rename({'S-Rp':'M'}).stack(stackDims)
# Convert to Pandas, use ep.multiDimXrToPD as a general multi-dimensional restacker
pd, _ = ep.multiDimXrToPD(basisPlot, colDims=stackDims)
# Summarise properties and tabulate via Pandas Describe
pd.describe().T
Show code cell output
count | mean | std | min | 25% | 50% | 75% | max | ||
---|---|---|---|---|---|---|---|---|---|
L | M | ||||||||
0 | 0 | 25.0 | 0.011 | 2.877e-01 | -0.282 | -0.282 | 0.282 | 0.282 | 0.282 |
1 | 1 | 32.0 | 0.005 | 2.265e-01 | -0.326 | -0.173 | 0.011 | 0.206 | 0.320 |
0 | 32.0 | 0.007 | 2.265e-01 | -0.238 | -0.222 | 0.011 | 0.221 | 0.282 | |
-1 | 32.0 | 0.005 | 2.265e-01 | -0.326 | -0.173 | 0.011 | 0.206 | 0.320 | |
2 | 2 | 34.0 | 0.021 | 2.047e-01 | -0.320 | -0.180 | 0.040 | 0.185 | 0.337 |
1 | 38.0 | 0.018 | 1.935e-01 | -0.261 | -0.162 | 0.059 | 0.216 | 0.282 | |
0 | 42.0 | 0.015 | 1.841e-01 | -0.229 | -0.180 | 0.057 | 0.167 | 0.282 | |
-1 | 38.0 | 0.018 | 1.935e-01 | -0.261 | -0.162 | 0.059 | 0.216 | 0.282 | |
-2 | 34.0 | 0.021 | 2.047e-01 | -0.320 | -0.180 | 0.040 | 0.185 | 0.337 | |
3 | 3 | 26.0 | 0.026 | 2.090e-01 | -0.326 | -0.140 | 0.077 | 0.188 | 0.320 |
2 | 32.0 | 0.031 | 1.865e-01 | -0.282 | -0.125 | 0.070 | 0.192 | 0.282 | |
1 | 38.0 | 0.032 | 1.701e-01 | -0.238 | -0.149 | 0.083 | 0.183 | 0.282 | |
0 | 38.0 | 0.033 | 1.700e-01 | -0.195 | -0.099 | -0.044 | 0.195 | 0.282 | |
-1 | 38.0 | 0.032 | 1.701e-01 | -0.238 | -0.149 | 0.083 | 0.183 | 0.282 | |
-2 | 32.0 | 0.031 | 1.865e-01 | -0.282 | -0.125 | 0.070 | 0.192 | 0.282 | |
-3 | 26.0 | 0.026 | 2.090e-01 | -0.326 | -0.140 | 0.077 | 0.188 | 0.320 | |
4 | 4 | 19.0 | 0.065 | 2.147e-01 | -0.229 | -0.166 | 0.107 | 0.248 | 0.337 |
3 | 26.0 | 0.054 | 1.825e-01 | -0.204 | -0.159 | 0.096 | 0.188 | 0.282 | |
2 | 33.0 | 0.046 | 1.621e-01 | -0.190 | -0.084 | 0.066 | 0.191 | 0.282 | |
1 | 36.0 | 0.044 | 1.547e-01 | -0.201 | -0.140 | 0.079 | 0.168 | 0.282 | |
0 | 39.0 | 0.042 | 1.486e-01 | -0.190 | -0.073 | 0.040 | 0.160 | 0.282 | |
-1 | 36.0 | 0.044 | 1.547e-01 | -0.201 | -0.140 | 0.079 | 0.168 | 0.282 | |
-2 | 33.0 | 0.046 | 1.621e-01 | -0.190 | -0.084 | 0.066 | 0.191 | 0.282 | |
-3 | 26.0 | 0.054 | 1.825e-01 | -0.204 | -0.159 | 0.096 | 0.188 | 0.282 | |
-4 | 19.0 | 0.065 | 2.147e-01 | -0.229 | -0.166 | 0.107 | 0.248 | 0.337 | |
5 | 5 | 10.0 | 0.112 | 2.433e-01 | -0.190 | -0.139 | 0.212 | 0.329 | 0.347 |
4 | 16.0 | 0.090 | 1.876e-01 | -0.208 | -0.015 | 0.145 | 0.232 | 0.295 | |
3 | 22.0 | 0.072 | 1.608e-01 | -0.196 | -0.069 | 0.127 | 0.199 | 0.260 | |
2 | 26.0 | 0.063 | 1.490e-01 | -0.179 | -0.065 | 0.085 | 0.190 | 0.246 | |
1 | 30.0 | 0.055 | 1.398e-01 | -0.161 | -0.092 | 0.093 | 0.173 | 0.241 | |
0 | 30.0 | 0.055 | 1.397e-01 | -0.171 | -0.076 | 0.054 | 0.166 | 0.246 | |
-1 | 30.0 | 0.055 | 1.398e-01 | -0.161 | -0.092 | 0.093 | 0.173 | 0.241 | |
-2 | 26.0 | 0.063 | 1.490e-01 | -0.179 | -0.065 | 0.085 | 0.190 | 0.246 | |
-3 | 22.0 | 0.072 | 1.608e-01 | -0.196 | -0.069 | 0.127 | 0.199 | 0.260 | |
-4 | 16.0 | 0.090 | 1.876e-01 | -0.208 | -0.015 | 0.145 | 0.232 | 0.295 | |
-5 | 10.0 | 0.112 | 2.433e-01 | -0.190 | -0.139 | 0.212 | 0.329 | 0.347 | |
6 | 6 | 6.0 | 0.160 | 2.560e-01 | -0.163 | -0.069 | 0.285 | 0.354 | 0.360 |
5 | 10.0 | 0.130 | 1.863e-01 | -0.200 | 0.100 | 0.204 | 0.255 | 0.289 | |
4 | 14.0 | 0.107 | 1.571e-01 | -0.191 | 0.004 | 0.166 | 0.230 | 0.266 | |
3 | 18.0 | 0.090 | 1.405e-01 | -0.156 | 0.048 | 0.109 | 0.179 | 0.252 | |
2 | 22.0 | 0.077 | 1.294e-01 | -0.165 | 0.004 | 0.090 | 0.177 | 0.243 | |
1 | 24.0 | 0.072 | 1.247e-01 | -0.160 | 0.007 | 0.099 | 0.152 | 0.230 | |
0 | 26.0 | 0.066 | 1.210e-01 | -0.156 | 0.007 | 0.062 | 0.174 | 0.239 | |
-1 | 24.0 | 0.072 | 1.247e-01 | -0.160 | 0.007 | 0.099 | 0.152 | 0.230 | |
-2 | 22.0 | 0.077 | 1.294e-01 | -0.165 | 0.004 | 0.090 | 0.177 | 0.243 | |
-3 | 18.0 | 0.090 | 1.405e-01 | -0.156 | 0.048 | 0.109 | 0.179 | 0.252 | |
-4 | 14.0 | 0.107 | 1.571e-01 | -0.191 | 0.004 | 0.166 | 0.230 | 0.266 | |
-5 | 10.0 | 0.130 | 1.863e-01 | -0.200 | 0.100 | 0.204 | 0.255 | 0.289 | |
-6 | 6.0 | 0.160 | 2.560e-01 | -0.163 | -0.069 | 0.285 | 0.354 | 0.360 | |
7 | 7 | 2.0 | 0.369 | 0.000e+00 | 0.369 | 0.369 | 0.369 | 0.369 | 0.369 |
6 | 4.0 | 0.260 | 2.159e-02 | 0.242 | 0.242 | 0.260 | 0.279 | 0.279 | |
5 | 6.0 | 0.208 | 5.293e-02 | 0.150 | 0.164 | 0.205 | 0.252 | 0.268 | |
4 | 8.0 | 0.174 | 6.729e-02 | 0.087 | 0.130 | 0.178 | 0.222 | 0.251 | |
3 | 10.0 | 0.149 | 7.566e-02 | 0.045 | 0.098 | 0.148 | 0.214 | 0.239 | |
2 | 12.0 | 0.129 | 8.089e-02 | 0.020 | 0.062 | 0.130 | 0.195 | 0.239 | |
1 | 14.0 | 0.114 | 8.427e-02 | 0.007 | 0.038 | 0.124 | 0.198 | 0.225 | |
0 | 14.0 | 0.114 | 8.308e-02 | 0.018 | 0.034 | 0.082 | 0.183 | 0.236 | |
-1 | 14.0 | 0.114 | 8.427e-02 | 0.007 | 0.038 | 0.124 | 0.198 | 0.225 | |
-2 | 12.0 | 0.129 | 8.089e-02 | 0.020 | 0.062 | 0.130 | 0.195 | 0.239 | |
-3 | 10.0 | 0.149 | 7.566e-02 | 0.045 | 0.098 | 0.148 | 0.214 | 0.239 | |
-4 | 8.0 | 0.174 | 6.729e-02 | 0.087 | 0.130 | 0.178 | 0.222 | 0.251 | |
-5 | 6.0 | 0.208 | 5.293e-02 | 0.150 | 0.164 | 0.205 | 0.252 | 0.268 | |
-6 | 4.0 | 0.260 | 2.159e-02 | 0.242 | 0.242 | 0.260 | 0.279 | 0.279 | |
-7 | 2.0 | 0.369 | 0.000e+00 | 0.369 | 0.369 | 0.369 | 0.369 | 0.369 | |
8 | 8 | 1.0 | 0.380 | NaN | 0.380 | 0.380 | 0.380 | 0.380 | 0.380 |
7 | 2.0 | 0.269 | 5.551e-17 | 0.269 | 0.269 | 0.269 | 0.269 | 0.269 | |
6 | 3.0 | 0.215 | 5.424e-02 | 0.184 | 0.184 | 0.184 | 0.231 | 0.277 | |
5 | 4.0 | 0.180 | 6.937e-02 | 0.120 | 0.120 | 0.180 | 0.240 | 0.240 | |
4 | 5.0 | 0.155 | 7.763e-02 | 0.075 | 0.075 | 0.189 | 0.189 | 0.249 | |
3 | 6.0 | 0.136 | 8.257e-02 | 0.043 | 0.066 | 0.136 | 0.205 | 0.228 | |
2 | 7.0 | 0.120 | 8.559e-02 | 0.022 | 0.056 | 0.090 | 0.188 | 0.238 | |
1 | 8.0 | 0.107 | 8.742e-02 | 0.010 | 0.042 | 0.097 | 0.161 | 0.222 | |
0 | 9.0 | 0.095 | 8.852e-02 | 0.003 | 0.027 | 0.094 | 0.188 | 0.234 | |
-1 | 8.0 | 0.107 | 8.742e-02 | 0.010 | 0.042 | 0.097 | 0.161 | 0.222 | |
-2 | 7.0 | 0.120 | 8.559e-02 | 0.022 | 0.056 | 0.090 | 0.188 | 0.238 | |
-3 | 6.0 | 0.136 | 8.257e-02 | 0.043 | 0.066 | 0.136 | 0.205 | 0.228 | |
-4 | 5.0 | 0.155 | 7.763e-02 | 0.075 | 0.075 | 0.189 | 0.189 | 0.249 | |
-5 | 4.0 | 0.180 | 6.937e-02 | 0.120 | 0.120 | 0.180 | 0.240 | 0.240 | |
-6 | 3.0 | 0.215 | 5.424e-02 | 0.184 | 0.184 | 0.184 | 0.231 | 0.277 | |
-7 | 2.0 | 0.269 | 5.551e-17 | 0.269 | 0.269 | 0.269 | 0.269 | 0.269 | |
-8 | 1.0 | 0.380 | NaN | 0.380 | 0.380 | 0.380 | 0.380 | 0.380 |
#*** Plot BLM terms for basis set - basic case
basisKey = 'BLMtableResort'
# Basic plot
ep.lmPlot(basisPlot, xDim=stackDims); # Basic plot with all terms
Show code cell output
Plotting data (No filename), pType=a, thres=0.01, with Seaborn
No artists with labels found to put in legend. Note that artists whose label start with an underscore are ignored when legend() is called with no argument.

#*** Plot BLM terms for basis set - plot with some additional figure formatting options
# Formatting options
titleString=f'$B_{{L,M}}$ terms for {sym}, lmax={lmaxPlot}'
titleDetails=True
labelRound = 1
catLegend=False
labelCols = [1,1]
# cmap = None for default.
# cmap = 'vlag'
# lmPlot with various options
*_, gFig = ep.lmPlot(basisPlot.where((basisPlot.l<=lmaxPlot)
& (basisPlot.lp<=lmaxPlot)),
xDim=stackDims, pType = 'r',
cmap=cmap, labelRound = labelRound,
catLegend=catLegend,
titleString=titleString, titleDetails=titleDetails,
labelCols = labelCols);
# Glue figure
glue("lmPlot_BLM_basis_D2h", gFig.fig, display=False)
Show code cell output
Plotting data (No filename), pType=r, thres=0.01, with Seaborn

Fig. 3.3 Example
3.3.6. Electric field geometric coupling term #
The coupling of two 1-photon terms (which arises in the square of the ionization matrix element as per Eq. (3.9)) can be written as a tensor contraction:
Where:
and define the field strengths for the polarizations and , which are coupled into the spherical tensor ;square-brackets indicate degeneracy terms, e.g.
;in the literature LF/AF field projection terms are usually denoted
(as above) or (used as a more general angular-momentum projection notation). For the MF case photon projection terms are usually denoted by or .The polarization vector or propagation direction is usually chosen to define the LF/AF z-axis for linear or non-linearly polarized light respectively (see Fig. 1.1 for the linear example), or defined relative to the molecular symmetry axis in the MF. (See Sect. 3.3.3 for further details.)
For a given case the polarization geometry may define a single projection term, or there may be multiple terms allowed. For instance, linearly polarized light is defined by
only, resulting in non-zero values for just the terms in Eq. (3.22), i.e. product terms are allowed. Non-linearly polarized light may contain all allowed components ( ). For the MF case, allowed components may be defined directly in the MF or determined via a frame transformation from the LF - see Sect. 3.3.7.Note this notation implicitly describes only the time-independent photon angular momentum coupling, but time-dependent/shaped laser fields can be readily incorporated by allowing for time-dependent fields
(see, for instance, Ref. [99]). (Support for this is planned in ePSproc [34], as of v1.3.2 this is in the development and testing phase.)
To derive this result, one can start from a general spherical tensor direct product, e.g., Eq. 5.36 in Zare [83]; for two first-rank tensors (e.g. electric field vectors) this contraction is given explicitly by Eq. 5.40 in Zare [83]:
This can be converted to
And the appropriate E-field terms substituted in to arrive at equation Eq. (3.22).
As before, we can visualise these values with the Photoelectron Metrology Toolkit [5], as illustrated in the following code block, and corresponding output Fig. 3.4.
#*** For illustration, recompute EPR term for default case.
EPRX = ep.geomCalc.EPR(form = 'xarray')
# Set parameters to restack the Xarray into (L,M) pairs
plotDimsRed = ['p', 'R-p']
xDim = {'PR':['P','R']}
# Plot with ep.lmPlot(), real values
*_, gFig = ep.lmPlot(EPRX, plotDims=plotDimsRed, xDim=xDim,
pType = 'r',
titleString=f'$E_{{P,R}}$ terms (all cases).',
labelCols = labelCols)
# Alternative version summed over l,l',m
# *_, gFig = ep.lmPlot(EPRX.unstack().sum(['l','lp','R-p']), xDim=xDim, pType = 'r')
# Glue figure
glue("lmPlot_EPR_basis", gFig.fig)
Show code cell output
Plotting data (No filename), pType=r, thres=0.01, with Seaborn

Fig. 3.4 Example
3.3.7. Molecular frame projection term #
For the molecular frame case, the coupling between the LF and MF can be defined by a projection term,
This is similar to the
For the LF/AF case, the same term appears but in a simplified form:
This form pertains since - in the LF/AF case - there is no specific frame transformation defined (i.e. there is no single molecular orientation defined in relation to the light polarization, rather a distribution as defined by the ADMs), but the total angular momentum coupling of the photon terms is still required in the equations.
Numerically, the function is calculated for a specified set of orientations, which default to the standard set of
3.3.8. Alignment tensor #
Finally, for the LF/AF case, the alignment tensor couples the molecular axis ensemble (defined as a set of ADMs, see Sect. 3.5 for details) and the photoionization multipole terms into the final observable.
In the full equations for the observable, this term appears in a summation with the ADMs, as:
This summed alignment term can be considered, essentially, as a (coherent) geometric averaging of the MF observable weighted by the axis distribution in the AF (for more on the axis averaging as a convolution, see Refs. [4, 96]); equivalently, the averaging can be considered as a purely angular-momentum coupling effect, which accounts for all contributing moments of the various aspects of the system, and defines the allowed projections onto the final observables in the LF.
Mappings of these terms are investigated numerically below, for some examplar cases.
3.3.8.1. Basic cases#
Fig. 3.5 illustrates the alignment tensor
For the simplest case of an unaligned ensemble, this term is restricted to
For circularly polarized light (
#*** Set range of ADMs for test as time-dependent values (linear ramps)
# Set ADMs for increasing alignment...
# Note that delta term is independent of the absolute values of the ADMs(t),
# but does use this term to define limits on some quantum numbers.
tPoints = 10 # 10 t-points
inputADMs = [[0,0,0, *np.ones(tPoints)],
[2,0,0, *np.linspace(0,1,tPoints)],
[4,0,0, *np.linspace(0,0.5,tPoints)],
[6,0,0, *np.linspace(0,0.3,tPoints)],
[8,0,0, *np.linspace(0,0.2,tPoints)]]
# Set to AKQS parameters in Xarray
AKQS = ep.setADMs(ADMs = inputADMs)
# Use default EPR term - note this computes for all pol states, p=[-1,0,1]
EPR = ep.geomCalc.EPR(form='xarray')
#*** Compute alignment terms
AFterm, DeltaTerm = ep.geomCalc.deltaLMKQS(EPR, AKQS)
#*** Plot Delta term with subselections and formatting
xDim = {'LM':['L','M']}
daPlot, daPlotpd, legendList, gFig = ep.lmPlot(
DeltaTerm.sel(K=0,Q=0,S=0,Rp=0).sel({'S-Rp':0}),
xDim = xDim,
pType = 'r', squeeze = False, thres=None,
titleString="$\Delta(0,0,0)$ term ($R'=0$ only)",
# Set dim mapping to use P,R with "l,m" colourmap
dimMaps={'lDims':['P'],'mDims':'R'},
labelCols=labelCols,
catLegend=False)
# Glue versions for JupyterBook output
glue("deltaTerm000-lmPlot", gFig.fig, display=False)
# As above, but with PD object return and glue.
glue("deltaTerm000-tab",daPlotpd.fillna(''), display=False)
Show code cell output
Set dataType (No dataType)
Plotting data (No filename), pType=r, thres=None, with Seaborn

Fig. 3.5 Example
L | 0 | 1 | 2 | |||||||
---|---|---|---|---|---|---|---|---|---|---|
M | 0 | -1 | 0 | 1 | -2 | -1 | 0 | 1 | 2 | |
P | R | |||||||||
0 | 0 | 1.0 | ||||||||
1 | -1 | -0.333 | ||||||||
0 | 0.333 | |||||||||
1 | -0.333 | |||||||||
2 | -2 | 0.2 | ||||||||
-1 | -0.2 | |||||||||
0 | 0.2 | |||||||||
1 | -0.2 | |||||||||
2 | 0.2 |
Fig. 3.6 Example
#*** Plot ADMs
*_, ADMFig = ep.lmPlot(AKQS, xDim = 't', pType = 'r',
labelCols=labelCols, catLegend=False,
cmap='vlag',
titleString='ADMs (linear ramp example)')
#*** Plot AF term with subselection
*_, AFFig = ep.lmPlot(AFterm.sel(R=0).sel(Rp=0).sel({'S-Rp':0}),
xDim = 't', pType = 'r',
labelCols=labelCols,
cmap='vlag',
titleString='$\\tilde{\Delta}_{L,M}(t)$ (linear ramp example)')
# Glue versions for JupyterBook output
glue("ADMs-linearRamp-lmPlot", ADMFig.fig, display=False)
glue("AFterm-linearRamp-lmPlot", AFFig.fig, display=False)
Show code cell output
Plotting data (No filename), pType=r, thres=0.01, with Seaborn
Set dataType (No dataType)
Plotting data (No filename), pType=r, thres=0.01, with Seaborn

Fig. 3.7 Example ADMs used for AF basis function example (see Fig. 3.8). These ADMs essentially show an increasing degree of alignment with the

Fig. 3.8 Example of
For cases with aligned molecular ensembles, additional terms can similarly appear depending on the alignment as well as the properties of the ionizing radiation. Again, the types of terms follow some typical patterns dependent on the symmetry of the ensemble, as well as the order of the terms allowed. For instance,
Since this is a coherent averaging, additional interferences can also appear in the AF - or be restricted in the AF - depending on these geometric parameters and the contributing matrix elements. Additionally, any effects modulating these terms, for instance a time-dependent alignment (rotational wavepacket), vibronic dynamics (vibrational and/or electronic wavepacket), time-dependent laser field (control field) may be anticipated to lead to both changes in these terms and, potentially, interesting effects in the observable. Such effects have been discussed in more detail in Quantum Metrology Vol. 2 [9], and in the current case the focus is purely on rotational wavepackets.
Fig. 3.8 shows
varies with alignment; in the demonstration case at later times, resulting in , whilst at , thus restricting terms to .Odd-
values are correlated with terms.Only
terms are allowed in this case ( ).
3.3.8.2. 3D alignments and symmetry breaking#
As discussed above, for the case where
#*** Set range of ADMs for test as time-dependent values (linear ramps),
# including some trial "3D" alignment terms
# Set ADMs for increasing alignment...
# Here add some (arb) terms for Q,S non-zero (indexed as [K,Q,S, ADMs(t)])
tPoints = 10
inputADMs3D = [[0,0,0, *np.ones(tPoints)],
[2,0,0, *np.linspace(0,1,tPoints)],
[2,0,2, *np.linspace(0,0.5,tPoints)],
[2,2,0, *np.linspace(0,0.5,tPoints)],
[2,2,2, *np.linspace(0,0.8,tPoints)]]
# Set to AKQS parameters in an Xarray
AKQS = ep.setADMs(ADMs = inputADMs3D)
# Compute alignment terms
AFterm, DeltaTerm = ep.geomCalc.deltaLMKQS(EPR, AKQS)
#*** Plot subsection, L<=2, and sum over Rp and S-Rp terms
titleString = ('$\\tilde{\Delta}_{L,M}(t)$ (3D alignment ramp example).'
'\n Summed over $R\'$ and $S-R\'$ terms.')
*_, gFig = ep.lmPlot(AFterm.where(AFterm.L<=2).sum('Rp').sum('S-Rp'),
xDim = 't', pType = 'r',
cmap='vlag',
titleString=titleString)
# Glue versions for JupyterBook output
glue("ADMs-3DlinearRamp-lmPlot", gFig.fig, display=False)
Show code cell output
Set dataType (No dataType)
Plotting data (No filename), pType=r, thres=0.01, with Seaborn

Fig. 3.9 Example
For illustration purposes, Fig. 3.9 shows a subselection of the
3.3.9. Tensor product terms#
Following the above, further resultant terms can also be examined, up to and including the full channel functions Xarray
[53, 54] data objects (for more details of the data structures used, see the ePSproc documentation [35], specifically the data structures page).
The main product basis returned, labelled polProd
in the output dictionary, contains the tensor product of the polarisation and alignment terms,
The full channel (response) functions
# Set data - set example ADMs to data structure & subset for calculation
data.setADMs(ADMs = inputADMs)
data.setSubset(dataKey = 'ADM', dataType = 'ADM')
# Using PEMtk - this only returns the product basis set as used for fitting
BetaNormX, basisProduct = data.afblmMatEfit(selDims={}, sqThres=False)
Subselected from dataset 'ADM', dataType 'ADM': 50 from 50 points (100.00%)
basisKey = 'polProd' # Key for BLM basis set
# Plot with subselection on pol state (by label, 'A'=z-pol case)
titleString=('"Polprod" basis term, $\Lambda_{R}\otimes E_{PR}(\hat{e})\otimes'
'\Delta_{L,M}(K,Q,S)\otimes A^{K}_{Q,S}(t)$.'
'\nSubselected for $z$-pol.')
*_, gFig = ep.lmPlot(basisProduct[basisKey].sel(Labels='A'),
xDim='t', cmap = cmap, mDimLabel='mu',
labelCols=labelCols,
titleString=titleString);
# Glue versions for JupyterBook output
glue("polProd-linearRamp-lmPlot", gFig.fig, display=False)
Show code cell output
Set dataType (No dataType)
Plotting data (No filename), pType=a, thres=0.01, with Seaborn
# Full channel functions
# Plot with subselection on pol state (by label, 'A'=z-pol case)
titleString='Channel functions example.\nSubselected for $z$-pol, $S-R\'=0$.'
*_, gFig = ep.lmPlot((basisProduct['BLMtableResort'] *
basisProduct['polProd']).sel(Labels='A').sel({'S-Rp':0}).sel(L=2),
xDim='t', cmap=cmap, mDimLabel='m',
titleString=titleString);
# Glue versions for JupyterBook output
glue("channelFunc-linearRamp-lmPlot", gFig.fig, display=False)
Show code cell output
Set dataType (No dataType)
Plotting data (No filename), pType=a, thres=0.01, with Seaborn
No artists with labels found to put in legend. Note that artists whose label start with an underscore are ignored when legend() is called with no argument.

Fig. 3.10 Example product basis function for the polarisation and ADM terms, as given by

Fig. 3.11 Example of