Source code for photutils.isophote.model

# Licensed under a 3-clause BSD style license - see LICENSE.rst
"""
Tools for building a model elliptical galaxy image from a list of
isophotes.
"""

import numpy as np
from scipy.interpolate import LSQUnivariateSpline

from photutils.utils._deprecation import deprecated_positional_kwargs

from .ellipse_model import build_ellipse_model_c

__all__ = ['build_ellipse_model']


[docs] @deprecated_positional_kwargs(since='3.0', until='4.0') def build_ellipse_model( shape, isolist, fill: float = 0.0, high_harmonics=False, sma_interval: float = 0.1, ): """ Build a model elliptical galaxy image from a list of isophotes. For each ellipse in the input isophote list the algorithm fills the output image array with the corresponding isophotal intensity. Pixels in the output array are in general only partially covered by the isophote "pixel". The algorithm takes care of this partial pixel coverage by keeping track of how much intensity was added to each pixel by storing the partial area information in an auxiliary array. The information in this array is then used to normalize the pixel intensities. Parameters ---------- shape : 2-tuple The (ny, nx) shape of the array used to generate the input ``isolist``. isolist : `~photutils.isophote.IsophoteList` instance The isophote list created by the `~photutils.isophote.Ellipse` class. fill : float, optional The constant value to fill empty pixels. If an output pixel has no contribution from any isophote, it will be assigned this value. The default is 0. high_harmonics : bool, optional Whether to add the higher-order harmonics (i.e., ``a3``, ``b3``, ``a4``, and ``b4``; see `~photutils.isophote.Isophote` for details) to the result. sma_interval : optional, float The interval between node values of the semi-major axis, which is used to spline interpolate values of other shape parameters. Returns ------- result : 2D `~numpy.ndarray` The image with the model galaxy. """ if len(isolist) == 0: msg = 'isolist must not be empty' raise ValueError(msg) # the target grid is spaced in 0.1 pixel intervals so as # to ensure no gaps will result on the output array. finely_spaced_sma = np.arange( isolist[0].sma, isolist[-1].sma, sma_interval, ) # interpolate ellipse parameters # End points must be discarded, but how many? # This seems to work so far nodes = isolist.sma[2:-2] intens_array = LSQUnivariateSpline( isolist.sma, isolist.intens, nodes)(finely_spaced_sma) eps_array = LSQUnivariateSpline( isolist.sma, isolist.eps, nodes)(finely_spaced_sma) pa_array = LSQUnivariateSpline( isolist.sma, isolist.pa, nodes)(finely_spaced_sma) x0_array = LSQUnivariateSpline( isolist.sma, isolist.x0, nodes)(finely_spaced_sma) y0_array = LSQUnivariateSpline( isolist.sma, isolist.y0, nodes)(finely_spaced_sma) grad_array = LSQUnivariateSpline( isolist.sma, isolist.grad, nodes)(finely_spaced_sma) if high_harmonics: a3_array = LSQUnivariateSpline( isolist.sma, isolist.a3, nodes)(finely_spaced_sma) b3_array = LSQUnivariateSpline( isolist.sma, isolist.b3, nodes)(finely_spaced_sma) a4_array = LSQUnivariateSpline( isolist.sma, isolist.a4, nodes)(finely_spaced_sma) b4_array = LSQUnivariateSpline( isolist.sma, isolist.b4, nodes)(finely_spaced_sma) grad_sma = -grad_array * finely_spaced_sma # Return deviations from ellipticity to their original amplitude # meaning kwargs_harm = { 'a3_array': a3_array * grad_sma, 'b3_array': b3_array * grad_sma, 'a4_array': a4_array * grad_sma, 'b4_array': b4_array * grad_sma, } else: kwargs_harm = {} # correct deviations caused by fluctuations in spline solution eps_array[np.where(eps_array < 0.0)] = 0.0 # for each interpolated isophote, generate intensity values on the # output image array result, weight = build_ellipse_model_c( shape[0], shape[1], finely_spaced_sma, intens_array, eps_array, pa_array, x0_array, y0_array, **kwargs_harm, ) # zero weight values must be set to 1.0 weight[np.where(weight <= 0.0)] = 1.0 # normalize result /= weight # fill value result[np.where(result == 0.0)] = fill return result