Source code for jdaviz.configs.default.plugins.gaussian_smooth.gaussian_smooth

import numpy as np

from astropy import units as u
from astropy.convolution import convolve, Gaussian2DKernel
from glue.core.message import (DataCollectionAddMessage,
                               DataCollectionDeleteMessage)
from specutils import Spectrum1D
from specutils.manipulation import gaussian_smooth
from spectral_cube import SpectralCube
from traitlets import List, Unicode, Int, Any, Bool, observe

from jdaviz.core.events import SnackbarMessage
from jdaviz.core.registries import tray_registry
from jdaviz.core.template_mixin import TemplateMixin
from jdaviz.utils import load_template

__all__ = ['GaussianSmooth']


spaxel = u.def_unit('spaxel', 1 * u.Unit(""))
u.add_enabled_units([spaxel])


[docs]@tray_registry('g-gaussian-smooth', label="Gaussian Smooth") class GaussianSmooth(TemplateMixin): template = load_template("gaussian_smooth.vue", __file__).tag(sync=True) stddev = Any().tag(sync=True) dc_items = List([]).tag(sync=True) selected_data = Unicode().tag(sync=True) show_modes = Bool(False).tag(sync=True) smooth_modes = List(["Spectral", "Spatial"]).tag(sync=True) selected_mode = Unicode("Spectral").tag(sync=True) def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) self.hub.subscribe(self, DataCollectionAddMessage, handler=self._on_data_updated) self.hub.subscribe(self, DataCollectionDeleteMessage, handler=self._on_data_updated) self._selected_data = None self._config = self.app.state.settings.get("configuration") if self._config == "cubeviz": self.show_modes = True def _on_data_updated(self, msg): self.dc_items = [x.label for x in self.data_collection] @observe("selected_data") def _on_data_selected(self, event): self._selected_data = next((x for x in self.data_collection if x.label == event['new']))
[docs] def vue_spectral_smooth(self, *args, **kwargs): # Testing inputs to make sure putting smoothed spectrum into # datacollection works # input_flux = Quantity(np.array([0.2, 0.3, 2.2, 0.3]), u.Jy) # input_spaxis = Quantity(np.array([1, 2, 3, 4]), u.micron) # spec1 = Spectrum1D(input_flux, spectral_axis=input_spaxis) size = float(self.stddev) try: spec = self._selected_data.get_object(cls=Spectrum1D) except TypeError: snackbar_message = SnackbarMessage( f"Unable to perform smoothing over selected data.", color="error", sender=self) self.hub.broadcast(snackbar_message) return # Takes the user input from the dialog (stddev) and uses it to # define a standard deviation for gaussian smoothing spec_smoothed = gaussian_smooth(spec, stddev=size) label = f"Smoothed {self._selected_data.label}" self.data_collection[label] = spec_smoothed snackbar_message = SnackbarMessage( f"Data set '{self._selected_data.label}' smoothed successfully.", color="success", sender=self) self.hub.broadcast(snackbar_message)
[docs] def vue_spatial_convolution(self, *args): """ Use astropy convolution machinery to smooth the spatial dimensions of the data cube. """ size = float(self.stddev) cube = self._selected_data.get_object(cls=SpectralCube) # Extend the 2D kernel to have a length 1 spectral dimension, so that # we can do "3d" convolution to the whole cube kernel = np.expand_dims(Gaussian2DKernel(size), 0) # TODO: in vuetify >2.3, timeout should be set to -1 to keep open # indefinitely snackbar_message = SnackbarMessage( "Smoothing spatial slices of cube...", loading=True, timeout=0, sender=self) self.hub.broadcast(snackbar_message) convolved_data = convolve(cube.hdu.data, kernel) # Create a new cube with the old metadata. Note that astropy # convolution generates values for masked (NaN) data, but we keep the # original mask here. newcube = SpectralCube(data=convolved_data, wcs=cube.wcs, mask=cube.mask, meta=cube.meta, fill_value=cube.fill_value) label = f"Smoothed {self._selected_data.label}" self.data_collection[label] = newcube snackbar_message = SnackbarMessage( f"Data set '{self._selected_data.label}' smoothed successfully.", color="success", sender=self) self.hub.broadcast(snackbar_message)