WFC3 Image Displayer & Analyzer#


Learning Goals#

This notebook provides a method to quickly display images from the Hubble Space Telescope’s Wide Field
Camera 3 (WFC3) instrument. This tool also allows the user to derive statistics by row or column in the image.

By the end of this tutorial, you will:

  • Download two WFC3 images from MAST.

  • Learn how to use the display_image tool to display any WFC3 fits file.

  • Learn how to use the row_column_stats tool to plot row or column statistics for any WFC3 fits file.

Table of Contents#

Introduction
1. Imports
2. Query MAST and download a WFC3 flt.fits and ima.fits image
3. display_image
      3.1 Display the full images with metadata
      3.2 Display UVIS1 & UVIS2 separately
      3.3 Display an image section and change the SCI array colormap
      3.4 Change the scaling of the SCI and ERR arrays
      3.5 Display each read of ima image section
4. row_column_stats
      4.1 Compute column median and column standard deviation for a full image
      4.2 Measure UVIS1 & UVIS2 column medians separately and overplot on one figure
      4.3 Display image section and compute column & row mean
      4.4 Compute column mean for each read of ima image section
5. Conclusions
Additional Resources
About the Notebook
Citations

Introduction#

One of the crucial steps in any analysis of WFC3 data is to visually inspect the
images. Whether you are viewing every image in the dataset or just a few
suspected outliers, displaying fits images takes time. To speed up the process
of displaying WFC3 images, we have created a tool in Python, display_image(),
that is meant to be used in a Jupyter Notebook. Once the function is imported
into your Jupyter Notebook session, you enter the WFC3 fits image you
want to display and it shows the full science, error, and data quality arrays
with a good scaling and colorbar. If the observation is a full frame UVIS
exposure, then the two CCDs are displayed together as one image (i.e. a 4kx4k
image). The display_image() tool allows for some customization, such as
displaying only a section of the image as well as changing the colormap or scaling.

In addition to visually inspecting WFC3 exposures, it can be helpful to compute
and plot statistics of the image as well. One common technique to quantify WFC3
images during analysis is to measure statistics of each individual column or row.
In this notebook we present a tool in Python, row_column_stats(), that is
meant to be used in a Jupyter Notebook. After importing the function into your
notebook session, you enter a WFC3 fits image along with the axis and
statistic you are interested in measuring. The function returns a plot of the
row (or column) statistic as a function of row (or column) number. In addition
to the plot, the function returns the row/column values and statistics being plotted.
This is useful if you want to plot multiple images on one figure. The
row_column_stats() tool allows for some customization, such as measuring
only a section of the image as well as choosing the axis (row or column) and the
statistic (mean, median, mode, or, standard deviation).

1. Imports#

This notebook assumes you have created and activated a virtual environment using the requirements file in this notebook's repository. Please make sure you have read the contents of the README file before continuing the notebook.

We import:

Package Name

Purpose

os

setting environment variables

astroquery.mast.Observations

downloading data from MAST

matplotlib.pyplot

plotting data

display_image

displaying any WFC3 image

row_column_stats

computing statistics on a WFC3 image

import os

from astroquery.mast import Observations
import matplotlib.pyplot as plt

from display_image import display_image
from row_column_stats import row_column_stats

%matplotlib inline

2. Query MAST and download a WFC3 flt.fits and ima.fits image#

You may download the data from MAST using either the Archive Search Engine or the MAST Portal.

Here, we download our images via astroquery. For more information, please look at the documentation for
Astroquery, Astroquery.mast, and CAOM Field Descriptions, which is used for the obs_table variable.

We download images of N2525 from proposal 15145, one being a flt.fits in WFC3/UVIS and the other
being a ima.fits in WFC3/IR. After downloading the images, we move them to our current working directory (cwd).

query = [('idgga5*', 'UVIS', 'FLT', (2, 3)),
         ('i*', 'IR', 'IMA', (10, 11))]

for criteria in query:
    # Get the observation records
    obs_table = Observations.query_criteria(obs_id=criteria[0], proposal_id=15145, target_name='N2525', instrument_name=f"WFC3/{criteria[1]}")

    # Get the listing of data products
    products = Observations.get_product_list(obs_table)

    # Filter the products for the file type
    filtered_products = Observations.filter_products(products, productSubGroupDescription=criteria[2], dataproduct_type='image')

    # Display the table
    filtered_products[criteria[3][0]:criteria[3][1]]

    # Download the images above
    download_table = Observations.download_products(filtered_products[criteria[3][0]:criteria[3][1]], mrp_only=False)

    # For convenience move file to cwd and remove empty download dir
    for file in download_table['Local Path']:
        filename = file.split('/')[-1]
        os.rename(file, os.path.basename(file))
        os.rmdir('mastDownload/HST/'+filename[:9])
        
    os.rmdir('mastDownload/HST/')
    os.rmdir('mastDownload/')
Downloading URL https://mast.stsci.edu/api/v0.1/Download/file?uri=mast:HST/product/idgga5m1q_flt.fits to ./mastDownload/HST/idgga5m1q/idgga5m1q_flt.fits ...
 [Done]
Downloading URL https://mast.stsci.edu/api/v0.1/Download/file?uri=mast:HST/product/idggabk1q_ima.fits to ./mastDownload/HST/idggabk1q/idggabk1q_ima.fits ...
 [Done]

List the fits files to verify they were moved properly

!ls -ltr *.fits
-rw-r--r-- 1 runner docker 168796800 Dec 31 16:48 idgga5m1q_flt.fits
-rw-r--r-- 1 runner docker 126201600 Dec 31 16:48 idggabk1q_ima.fits

3. display_image#

In this section, we demonstrate the functionality of display_image, a useful tool for quickly analyzing WFC3 images.
The subsections explain how to display full images with metadata, individual WFC3/UVIS chip images, a section of an
image with various colormaps/scaling, and individual WFC3/IR ima reads.

3.1 Display the full images with metadata#

First, we display the SCI, ERR, and DQ arrays for each image and print header info. The default value for printmeta
is False. In the cell below, we set the keyword printmeta to True to print useful information from the header of the
file to the screen. The WFC3/UVIS image is in electrons and the WFC3/IR image is in electrons/second. See Section 2.2.3
of the WFC3 Data Handbook for full descriptions of SCI, ERR, and DQ arrays.

display_image('idgga5m1q_flt.fits', printmeta=True)
display_image('idggabk1q_ima.fits', printmeta=True)
	WFC3/UVIS idgga5m1q_flt.fits 
--------------------------------------------
Filter = F350LP, Date-Obs = 2018-03-04 T21:11:43,
Target = N2525, Exptime = 355.0, Subarray = False, Units = ELECTRONS
	WFC3/IR idggabk1q_ima.fits 
--------------------------------------------
Filter = F160W, Date-Obs = 2018-09-11 T18:16:13,
Target = N2525, Exptime = 502.936462, Subarray = False, Units = ELECTRONS/S
../../../_images/2702e9ba6c281e9423dcb36779d2f5ba70f349c985286ae1c0d84e7b74e612aa.png ../../../_images/c65b765e66d1bb7eeb8cd1d2c6d9288f5458dcd42de9bed23f08178424b533ee.png

3.2 Display UVIS1 & UVIS2 separately#

Next, we display the WFC3/UVIS chips separately. To select a section of an image, append [xstart:xend,ystart:yend] to the
image name as shown below. Notice how we need to specify the full axis range [0:4096,0:2051] and not simply just
[:,:2051] as in standard numpy notation.

print('Display UVIS1')
display_image('idgga5m1q_flt.fits[0:4096,2051:4102]') 
Display UVIS1
../../../_images/f299c20865c55a85bfc7c7b13ca119c1dfa35b407da08be3b3dd516e885a5cd2.png
print('Display UVIS2')
display_image('idgga5m1q_flt.fits[0:4096,0:2051]') 
Display UVIS2
../../../_images/9a25bf42c5932a845e26ee9a3997658b5da94d1acbe33fa725e34d0786643882.png

3.3 Display an image section and change the SCI array colormap#

Then, we display SCI arrays with a different colormap. Regardless of how many colormaps are being changed, all three
colormaps must be provided. The elements of colormaps sequentially correspond with the SCI, ERR, and DQ arrays.

display_image('idgga5m1q_flt.fits[3420:3575,2590:2770]',
              colormaps=["viridis", "Greys_r", "inferno_r"])
../../../_images/35d6cdbca810928912ad3e16f8f03b5108a51d919440e92c76f4ea21d5bf4017.png

3.4 Change the scaling of the SCI and ERR arrays#

Now, we change the scaling of the SCI and ERR arrays. Regardless of how many scalings are being changed, all three
scalings must be provided. The elements of scaling sequentially correspond with the SCI, ERR, and DQ arrays.
The default scaling value of None uses astropy.visualization.ZScaleInterval() for scaling
(see documentation for more information).

display_image('idgga5m1q_flt.fits[3420:3575,2590:2770]',
              colormaps=["viridis", "viridis", "inferno_r"],
              scaling=[(50000, 80000), (None, 400), (None, None)])
../../../_images/cac2cb0a3eeaad642b73f3f147fd516d6e80b1635a05a33626590bf53b0cfd75.png

3.5 Display each read of ima image section#

In addition to changing each SCI array’s colormap and scaling, we display each read of the ima by setting ima_multiread=True.

display_image('idggabk1q_ima.fits[43:55,299:311]',
              colormaps=["viridis", "Greys_r", "inferno_r"],
              scaling=[(2, 18), (None, None), (None, None)],
              ima_multiread=True)
../../../_images/84bc4ff9d44e7a5aa28c2a82bb97e7c830bd98e977a165da06452500ff3477a3.png ../../../_images/23fc550f46dda322dc969686324210d56031c1ca078f8442458898963165e845.png ../../../_images/d196f03f2dec3ba7d3d1a99ac66341c3b492021f6f06c411e7927e1e0d82da36.png ../../../_images/ee19673c24001a91467e89e3e0f1d53427fbe22f2edc2e9bc594bb65591a8cc8.png ../../../_images/06afac3d50a1586409295f8eceda97e6fcdf37ac211fd57f8ee8959167a4df1f.png ../../../_images/7317106dbbb1fba84504267a75e3c5ee2c81a84988b00ea343443dbed6fabb98.png ../../../_images/43d54abb2dcd6b2d3fba9743aec752d0934fe758bad3ff950659b8ffa7704b42.png ../../../_images/1cff2a5863f8eb1f96a9f0dfa7dd0525bbbf888a287e1537bbaf1b1c0b644d98.png ../../../_images/a1f4cd046eb147cb15d60d521b8d81937be8cbf872037868aee9cf40ee7a42e3.png ../../../_images/d928b319486fa53b311bd1694cfd0d8a166d067c0f0a3ca0821b6f21fe41f961.png ../../../_images/4627cb48ba7976393dde5ef93d3dd74c09439614eb263ba1ac570cf6ebdc7b4f.png ../../../_images/44ef598cb71fb8490a67c537f46f09b1a3d4ca6b1d71a35978b564dec6621557.png

4. row_column_stats#

In this section, we demonstrate the functionality of row_column_stats, a useful tool for quickly computing WFC3 statistics.
The subsections explain how to compute row and column statistics for a full image, individual WFC3/UVIS chips, a section of
an image, and individual ima reads. The row/column numbers are on the x-axis and the statistics are on the y-axis.

4.1 Compute column median and column standard deviation for a full image#

First, we plot the column median and standard deviations for our WFC3/UVIS image.

# plot column median for the full image
x, y = row_column_stats('idgga5m1q_flt.fits',
                        stat='median',
                        axis='column')

# plot column standard deviation for the full image
x, y = row_column_stats('idgga5m1q_flt.fits',
                        stat='stddev',
                        axis='column')
../../../_images/b7f557fd67e11ec8ac190ba7dc394b18d7e712c3612dc3dde5b9ca46540970bc.png ../../../_images/a900f97cf1355723857304b003763a4dae124d6e5abd4273d078c4b319431000.png

4.2 Measure UVIS1 & UVIS2 column medians separately and overplot on one figure#

Next, we plot the column medians for each individual WFC3/UVIS chip.

# get column median values for UVIS2 but don't plot 
x2, y2 = row_column_stats('idgga5m1q_flt.fits[0:4096,0:2051]',
                          stat='median',
                          axis='column',
                          plot=False)

# get column median values for UVIS1 but don't plot 
x1, y1 = row_column_stats('idgga5m1q_flt.fits[0:4096,2051:4102]',
                          stat='median',
                          axis='column',
                          plot=False)

# overplot UVIS1 and UVIS2 data on one figure 
plt.figure(figsize=(8, 6), dpi=130)
plt.grid(alpha=.5)
plt.plot(x1, y1, marker='.', label='UVIS 1', color='k')
plt.plot(x2, y2, marker='.', label='UVIS 2', color='C3')
plt.title('WFC3/UVIS idgga5m1q_flt.fits')
plt.xlabel('Column Number')
plt.ylabel('Column Median [e-]')
plt.legend()
<matplotlib.legend.Legend at 0x7f193d281810>
../../../_images/0b8801aac1a645267c7b89147550c2ef88facfd364c665212f1ac5d6d6d01a29.png

4.3 Display image section and compute row & column mean#

Now, we compute the row and column means for a section of our WFC3/UVIS image.

# Display a section of the image
display_image('idgga5m1q_flt.fits[3420:3575,2590:2770]')

# plot row mean for a section of the image
x, y = row_column_stats('idgga5m1q_flt.fits[3420:3575,2590:2770]',
                        stat='mean',
                        axis='row')

# plot column mean for a section of the image
x, y = row_column_stats('idgga5m1q_flt.fits[3420:3575,2590:2770]',
                        stat='mean',
                        axis='column')
../../../_images/b746d467b5422a7f160fe8e3c4a410bbd45e95ece641733a8745af00e04de0f2.png ../../../_images/2de54848599f70b0d653ab38fdd7614ed71dbf3f0e278f9e3795571bf2091199.png ../../../_images/fc1df02be802191a6334fcfce3c4b8eb50591f0cf2ded99c1c3d0ff9365bb91e.png

Note: Without specifying ylim, the y-axis limits (above) are set to matplotlib defaults.
Set ylim to (y1,y2) (below) to create custom y-axis limits from the previous example.

y1 = 50
y2 = 220

# Display a section of the image
display_image('idgga5m1q_flt.fits[3420:3575,2590:2770]')

# plot row mean for single source with custom yaxis limits
x, y = row_column_stats('idgga5m1q_flt.fits[3420:3575,2590:2770]',
                        stat='mean',
                        axis='row',
                        ylim=(y1, y2))

# plot column mean for single source with custom yaxis limits
x, y = row_column_stats('idgga5m1q_flt.fits[3420:3575,2590:2770]',
                        stat='mean',
                        axis='column',
                        ylim=(y1, y2))
../../../_images/b746d467b5422a7f160fe8e3c4a410bbd45e95ece641733a8745af00e04de0f2.png ../../../_images/a2af34510f349e9b3f075c205df090158d42ed7df7c95a57d705ed6a3e077833.png ../../../_images/b1d8b35593f3356ff00da23c7e671b1c1b43f91216abb9ecb6de996560c6b4a8.png

4.4 Compute column mean for each read of ima image section#

Finally, we compute the column means for the same section of each ima read.

# Display a section of the image
display_image('idggabk1q_ima.fits[43:55,299:311]')

# plot column mean for section of ima
x, y = row_column_stats('idggabk1q_ima.fits[43:55,299:311]',
                        stat='mean',
                        axis='column',
                        ima_multiread=True)
../../../_images/8482affe6146f1bc619101421e03de7e95fb43451cb2fc1246ba39b803fc77cd.png ../../../_images/9061c0352ad141800162f427bfa88115ef61a46fe6189cc92cd08efade94f534.png ../../../_images/21f31d037bb75398ca65fb7add4d3c7bf2a2de85990900d3dd57ef18dc95e31e.png ../../../_images/0d70ac51e675418b7fd70f5340dcb3ea98c52565fb84f9e321dfb261604a57fa.png ../../../_images/63379ab2e48561d3529151290920f082f535c2ee7f75d49d2f70f4551acb9c7c.png ../../../_images/5d8d982607be18ebf52844f80e10618318b5fd6a0f0e1dd76dc8256329524fdf.png ../../../_images/91b6ffd0666124cdeee87a15911275c1b8fe9191fa58b7fe6e16864855e62ed0.png ../../../_images/371f5eabeaf560c9b9b92af8013b02db59ba97bdc5d431d8c34973ba280db943.png ../../../_images/badf15de1fe2ef562adbd41d39eb975f49f1afc84cc8c3d6b6ef1d08a2a102d7.png ../../../_images/b480274967d24770e50781dcefc12ca150d8d8d8be83c6377dadf7790b24a356.png ../../../_images/3f6c81a9a58368c3cdb4fcb339184502dd6589178c799dd28a4f41ece32838ce.png ../../../_images/1e5c850ce5baa4368acdfe2631b3e36f67061939e3e758d00c334665e43e4454.png ../../../_images/bcf29b854de5a45304cef4330c0240c7e5f72e10f332553d1959d9fa0827c8d3.png

5. Conclusions#

Thank you for walking through this notebook. Now while analyzing WFC3 data, you should be more familiar with:

  • Using display_image to quickly inspect the SCI, ERR, and DQ arrays of an image or image section.

  • Using row_column_stats to compute statistics on each individual column (or row) of an image or image section.

Congratulations, you have completed the notebook!#

Additional Resources#

Below are some additional resources that may be helpful. Please send any questions through the HST Helpdesk.

About this Notebook#

Author: Benjamin Kuhn; WFC3 Instrument Team

Updated on: 2021-10-04

Citations#

If you use numpy, matplotlib, astropy, or astroquery for published research, please cite the authors.
Follow these links for more information about citing the libraries below:


Top of Page Space Telescope Logo