https://github.com/sevenian3/ChromaStarPy
Raw File
Tip revision: 103d3d0df6d9574c49818f149e1cae8100455d10 authored by Ian Short on 06 July 2023, 18:09:20 UTC
Create ReadMe
Tip revision: 103d3d0
PostProcess.py
# -*- coding: utf-8 -*-
"""
Created on Wed May  3 12:20:48 2017

@author: ishort
"""

"""/**** Routines for client side post-processing of raw model atmosphere/spectrum
 * synthesis output from server to produce synthetic observables
 */"""


import math
import numpy
import Useful
import ToolBox
nm2cm = 1.0e-7

def UBVRIraw(lambdaScale, flux):

    """
    #Computes raw UBVRI fluxes
    """
    
    filters = filterSet()

    numBands = len(filters)
    #var numLambdaFilt

    bandFlux = [0.0 for i in range(numBands)]


    #var deltaLam, newY, product;

    for ib in range(numBands):

        bandFlux[ib] = 0.0 #//initialization
        numLambdaFilt = len(filters[ib][0])
        #//console.log("ib " + ib + " numLambdaFilt " + numLambdaFilt);
        #//wavelength loop is over photometric filter data wavelengths

        for il in range(1, numLambdaFilt):

            #//In this case - interpolate model SED onto wavelength grid of given photometric filter data

            deltaLam = filters[ib][0][il] - filters[ib][0][il - 1]  #//nm
            #//deltaLam = 1.0e-7 * deltaLam;  //cm
            #//console.log("ib: " + ib + " il: " + il + " filters[ib][0][il] " + filters[ib][0][il] + " deltaLam: " + deltaLam + " filters[ib][1][il] " + filters[ib][1][il]);

            #//hand log flux (row 1) to interpolation routine: 
            newY = ToolBox.interpol(lambdaScale, flux[1], filters[ib][0][il])
            #// linearize interpolated flux: - fluxes add *linearly*
            newY = math.exp(newY)

            product = filters[ib][1][il] * newY
            #if (ib == 2):
            #    //console.log("Photometry: il: " + il + " newY: " + newY + " filterLamb: " + filters[ib][0][il] + " filterTrans: " + filters[ib][1][il] + " product " + product);
            
            #//System.out.println("Photometry: filtertrans: " + filters[ib][1][il] + " product: " + product + " deltaLam: " + deltaLam);
            #//Rectangular picket integration
            bandFlux[ib] = bandFlux[ib] + (product * deltaLam)
            #//console.log("Photometry: ib: " + ib + " deltaLam " + deltaLam + " bandFlux: " + bandFlux[ib]);

        #} //il loop - lambdas
        #//console.log("Photometry: ib: " + ib + " bandFlux: " + bandFlux[ib], " product " + product + " deltaLam " + deltaLam);

    #}  //ib loop - bands

    #var raw;

    return bandFlux

#}; //UBVRI

#//
#//
    
def UBVRI(bandFlux):

    """/**
 * Computes colors from band-integrated fluxes from UBVRIraw()
 * First, reality check raw colours, THEN Run Vega model and subtract off Vega
 * colours for single-point calibration
 */"""    

    #Must be consistent with UBVRIraw()
    #As of April 2020:  Index, band
    #                   0,     Ux
    #                   1,     Bx
    #                   2,     B
    #                   3,     V
    #                   4,     R
    #                   5,     I
    #                   6,     H
    #                   7,     J
    #                   8,     K

    numCols = 7  #//seven band combinations in Johnson-Bessell UxBxBVRI: Ux-Bx, B-V, V-R, V-I, R-I, V-K, J-K
    colors = [0.0 for i in range(numCols)]

    #// Single-point calibration to Vega:
    #// Vega colours computed self-consistently using 
    #// Stellar parameters of Castelli, F.; Kurucz, R. L., 1994, A&A, 281, 817
    #// Teff = 9550 K, log(g) = 3.95, [Fe/H] = -0.5:

    #vegaColors = [0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0]; #//For re-calibrating with raw Vega colours
    #// Aug 2015 - with 14-line linelist:
    #//var vegaColors = [0.289244, -0.400324, 0.222397, -0.288568, -0.510965];
    #//var vegaColors = [0.163003, -0.491341, 0.161940, -0.464265, -0.626204];
    #//With Balmer line linear Stark broadening wings:
    #//vegaColors = [0.321691, -0.248000, 0.061419, -0.463083, -0.524502];
    #vegaColors = [0.53, -0.66, 0.11, -0.51, -0.62]
    vegaColors = [0.0528, -0.6093, 0.3241, -1.1729, -0.6501, -3.2167, -1.5526] #//lburns, June 2017

    #// Ux-Bx: 
    raw = 2.5 * math.log10(bandFlux[1] / bandFlux[0])
    colors[0] = raw - vegaColors[0]
    #//console.log("U-B: " + colors[0] + " raw " + raw + " bandFlux[1] " + bandFlux[1] + " bandFlux[0] " + bandFlux[0]);

    #// B-V:
    raw = 2.5 * math.log10(bandFlux[3] / bandFlux[2])
    colors[1] = raw - vegaColors[1]
    #//console.log("B-V: " + colors[1])

    #// V-R:
    raw = 2.5 * math.log(bandFlux[4] / bandFlux[3])
    colors[2] = raw - vegaColors[2]
    #//console.log("V-R: " + colors[2]);

    #// V-I:
    raw = 2.5 * math.log(bandFlux[5] / bandFlux[3])
    colors[3] = raw - vegaColors[3]
    #//console.log("V-I: " + colors[3]);

    #// R-I:
    raw = 2.5 * math.log10(bandFlux[5] / bandFlux[4])
    colors[4] = raw - vegaColors[4]
    #//console.log("R-I: " + colors[4]);
    
    #// V-K: lburns
    raw = 2.5 * math.log10(bandFlux[8] / bandFlux[3]);
    colors[5] = raw - vegaColors[5];
    #//console.log("V-K: " + colors[5]);

    #// J-K: lburns
    raw = 2.5 * math.log10(bandFlux[8] / bandFlux[7]);
    colors[6] = raw - vegaColors[6];
    #//console.log("J-K: " + colors[6]);


    return colors

#}; //UBVRI

#//
#//

def iColors(lambdaScale, intens, numThetas, numLams):
    #//No! iColors now returns band-integrated intensities

    filters = filterSet()
    numCols = 7 #//five band combinations in Johnson-Bessell UxBxBVRI: Ux-Bx, B-V, V-R, V-I, R-I

    numBands = len(filters)
    numLambdaFilt = len(filters[0][0])
 

    #//var colors = [];
    #//colors.length = numCols;
    #//// Have to use Array constructor here:
    #//for (var i = 0; i < numCols; i++) {
    #//    colors[i] = new Array(numThetas);
    #//}


    bandIntens = [ [ 0.0 for i in range(numThetas) ] for j in range(numBands) ]
    
    #JS: #// Have to use Array constructor here:
    #for i in range(numBands):
    #    bandIntens[i] = []
    #    bandIntens[i].length = numThetas;
    #}


    #//Unnecessary:
    #//Note: Calibration must be re-done!  May 2015
    #// Single-point Johnson UBVRI calibration to Vega:
    #// Vega colours computed self-consistntly with GrayFox 1.0 using 
    #// Stellar parameters of Castelli, F.; Kurucz, R. L., 1994, A&A, 281, 817
    #// Teff = 9550 K, log(g) = 3.95, ([Fe/H] = -0.5 - not directly relevent):
    vegaColors = [0.163003, -0.491341, 0.161940, -0.464265, -0.626204]

    #var deltaLam, newY, product, raw;
    intensLam = [0.0 for i in range(numLams)]
    

    #//Now same for each intensity spectrum:
    for it in range(numThetas):

        #//Caution: This loop is over *model SED* lambdas!
        for jl in range(numLams):
            intensLam[jl] = intens[jl][it]
            #//System.out.println("it: " + it + " jl: " + jl + " intensLam[jl]: " + intensLam[jl]);
        

        for ib in range(numBands):

            bandIntens[ib][it] = 0.0 #//initialization

            #//wavelength loop is over photometric filter data wavelengths

            for il in range(1, numLambdaFilt):

                #//In this case - interpolate model SED onto wavelength grid of given photometric filter data

                deltaLam = filters[ib][0][il] - filters[ib][0][il - 1] #//nm
                #//deltaLam = 1.0e-7 * deltaLam; //cm
                #//hand log flux (row 1) to interpolation routine: 
                newY = ToolBox.interpol(lambdaScale, intensLam, filters[ib][0][il])
                #//System.out.println("Photometry: newFlux: " + newFlux + " filterlamb: " + filters[ib][0][il]);
                product = filters[ib][1][il] * newY
                #//System.out.println("Photometry: filtertrans: " + filters[ib][1][il] + " product: " + product + " deltaLam: " + deltaLam);
                #//Rectangular picket integration
                bandIntens[ib][it] = bandIntens[ib][it] + (product * deltaLam)
                #//console.log("Photometry: ib: " + ib + " bandIntens: " + bandIntens[ib][it]);
            #}  //il wavelength loop

        #//System.out.println("Photometry: ib: " + ib + " it: " + it + " bandIntens: " + bandIntens[ib][it]);
        #}  //ib band loop

        #//necessary
        #//Make the colors! :-)
        #//console.log("it: " + it);

        #// Ux-Bx: 
        #//raw = 2.5 * logTen(bandIntens[1][it] / bandIntens[0][it]);
        #//colors[0][it] = raw - vegaColors[0];
        #//console.log("U-B: " + colors[0][it]);

        #// B-V:
        #//raw = 2.5 * logTen(bandIntens[3][it] / bandIntens[2][it]);
        #//colors[1][it] = raw - vegaColors[1];
        #//console.log("B-V: " + colors[1][it]);

        #// V-R:
        #//raw = 2.5 * logTen(bandIntens[4][it] / bandIntens[3][it]);
        #//colors[2][it] = raw - vegaColors[2];
        #//console.log("V-R: " + colors[2][it]);

        #// V-I:
        #// raw = 2.5 * logTen(bandIntens[5][it] / bandIntens[3][it]);
        #//colors[3][it] = raw - vegaColors[3];
        #//console.log("V-I: " + colors[3][it]);

        #// R-I:
        #//raw = 2.5 * logTen(bandIntens[5][it] / bandIntens[4][it]);
        #//colors[4][it] = raw - vegaColors[4];
        #//console.log("R-I: " + colors[4][it]);
        #//necessary

    #} //theta it loop

    #//return colors;
    return bandIntens

#}; //iColours


#//Create area normalized Gaussian appropriate for interpolating onto high resolution wavelength gid
def gaussian(lambdaScale, numLams, lambdaIn, sigmaIn, lamUV, lamIR):
    #//No! iColors now returns band-integrated intensities

    #//diskSigma = 10.0;  //test
    #//diskSigma = 0.01;  //test
    #//wavelength sampling interval in nm for interpolation 
    deltaLam = 0.001  #//nm
    sigma = sigmaIn / deltaLam #//sigma of Gaussian in pixels 
    #//Number of wavelength elements for Gaussian
    numSigmas = 2.5 #//+/- 2.5 sigmas
    numGauss = int(math.ceil(2.0 * numSigmas * sigma))  #//+/- 2.5 sigmas
    #//ensure odd number of elements in Gausian kernal
    if ((numGauss % 2) == 0):
        numGauss+=1
       
    #//Row 0 holds wavelengths in cm
    #//Row 1 holds Gaussian
    gauss = [ [ 0.0 for i in range(numGauss) ] for j in range(2) ]
    #jsgauss.length = 2;
    #gauss[0] = [];
    #gauss[0].length = numGauss;
    #gauss[1] = [];
    #gauss[1].length = numGauss;
    midPix = math.floor(numGauss/2)

    #////Area normalization factors:
    #//       var rootTwoPi = Math.sqrt(2.0 * Math.PI);
    #//       var prefac = 1.0 / (sigma * rootTwoPi); 

    #var x, expFac;
    #//Construct Gaussian in pixel space:
    #//var sum = 0.0;  //test
    for i in range(numGauss):
        x = (i - midPix)
        expFac = x / sigma 
        expFac = expFac * expFac 
        gauss[1][i] = math.exp(-0.5 * expFac) 
        #//gauss[i] = prefac * gauss[i];
        #//sum+= gauss[i];   //test
        #//console.log("i " + i + " gauss[i] " + gauss[i]);  //test
         
    #//console.log("Gaussian area: " + sum); 

    #//establish filter lambda scale:
    #//var filterLam = [];
    #//filterLam.length = numGauss;
    lamStart = lambdaIn - (numSigmas * sigmaIn) #//nm
    ii = 0; 
    for i in range(numGauss):
        ii = 1.0 * i
        #// filterLam[i] = lamStart + (ii * deltaLam); //nm
        #// filterLam[i] = 1.0e-7 * filterLam[i]; //cm for reference to lambdaScale
        gauss[0][i] = lamStart + (ii * deltaLam) #//nm
        gauss[0][i] = nm2cm * gauss[0][i] #//cm for reference to lambdaScale
        #//Keep wthin limits of treated SED:
        if (gauss[0][i] < nm2cm*lamUV):
            gauss[1][i] = 0.0
        #}  
        if (gauss[0][i] > nm2cm*lamIR):
            gauss[1][i] = 0.0
        #}  
    #} 
    #//for (var i = 0; i < numGauss; i++){
    #//console.log("i " + i + " filterLam[i] " + filterLam[i] + " gauss[i] " + gauss[i]);  //test
    #//  } 

    #//console.log("numGauss " + numGauss + " sigma " + sigma + " lamStart " + lamStart);

    return gauss

#}; //gaussian


#//var tuneColor = function(lambdaScale, intens, numThetas, numLams, diskLambda, diskSigma, lamUV, lamIR) {
def tuneColor(lambdaScale, intens, numThetas, numLams, gaussian, lamUV, lamIR):
    #//No! iColors now returns band-integrated intensities

    #//diskSigma = 10.0;  //test
    #//diskSigma = 0.01;  //test
    #//wavelength sampling interval in nm for interpolation 

    numGauss = len(gaussian[0])
    deltaLam = gaussian[0][1] - gaussian[0][0]

    bandIntens = [0.0 for i in range(numThetas)]
    
    #var product;
    intensLam = [0.0 for i in range(numLams)]

    #//Now same for each intensity spectrum:
    for it in range(numThetas):

        #//Caution: This loop is over *model SED* lambdas!
        for jl in range(numLams):
            intensLam[jl] = intens[jl][it]
            #//System.out.println("it: " + it + " jl: " + jl + " intensLam[jl]: " + intensLam[jl]);
        

    #//    for (var ib = 0; ib < numBands; ib++) {

        bandIntens[it] = 0.0 #//initialization
        #//var newY = interpolV(intensLam, lambdaScale, filterLam);
        #newY = ToolBox.interpolV(intensLam, lambdaScale, gaussian[0])
        newY = numpy.interp(gaussian[0], lambdaScale, intensLam)

        #//wavelength loop is over photometric filter data wavelengths

        for il in range(1, numGauss):

            #//In this case - interpolate model SED onto wavelength grid of tunable filter 

            #//product = gauss[il] * newY[il];
            product = gaussian[1][il] * newY[il]
            #//Rectangular picket integration
            bandIntens[it] = bandIntens[it] + (product * deltaLam)
            #//console.log("Photometry: ib: " + ib + " bandIntens: " + bandIntens[ib][it]);
        #} //il wavelength loop

        #//console.log("tuneColor: it: " + it + " bandIntens: " + bandIntens[it]);
        #//   }  //ib band loop

    #} //theta it loop

    #//return colors;
    return bandIntens

#}; //tuneColor

#//
#//

def filterSet():

    numBands = 9 #// Bessell-Johnson UxBxBVRI
    numLambdaFilt = 25 #//test for now

    #//double[][][] filterCurves = new double[numBands][2][numLambdaFilt];

    filterCurves = [  [  [ 0.0 for i in range(numLambdaFilt) ] for j in range(2) ] for k in range(numBands) ]
    #JS: filterCurves.length = numBands;
    #// Have to use Array constructor here:
    #for (var i = 0; i < numBands; i++) {
    #    filterCurves[i] = [];
    #    filterCurves[i].length = 2;
    #}
    #// Have to use Array constructor here:
    #for (var i = 0; i < numBands; i++) {
    #    filterCurves[i][0] = [];
    #    filterCurves[i][1] = [];
    #    filterCurves[i][0].length = numLambdaFilt;
    #    filterCurves[i][1].length = numLambdaFilt;
    #}

    #//Initialize all filterCurves - the real data below won't fill in all the array elements:
    for ib in range(numBands):
        for il in range(numLambdaFilt):
            filterCurves[ib][0][il] = 1000.0 #//placeholder wavelength (nm)
            filterCurves[ib][1][il] = 0.0e0 #// initialize filter transparency to 0.0
        #}
    #}

    #//http://ulisse.pd.astro.it/Astro/ADPS/Systems/Sys_136/index_136.html
    #//Bessell, M. S., 1990, PASP, 102, 1181
    #//photometric filter data for Bessell UxBxBVRI system from Asiago database in Java & JavaScript syntax
    #//Individual bands are below master table
    #//        UX
    filterCurves[0][0][0] = 300.0;
    filterCurves[0][1][0] = 0.000;
    filterCurves[0][0][1] = 305.0;
    filterCurves[0][1][1] = 0.016;
    filterCurves[0][0][2] = 310.0;
    filterCurves[0][1][2] = 0.068;
    filterCurves[0][0][3] = 315.0;
    filterCurves[0][1][3] = 0.167;
    filterCurves[0][0][4] = 320.0;
    filterCurves[0][1][4] = 0.287;
    filterCurves[0][0][5] = 325.0;
    filterCurves[0][1][5] = 0.423;
    filterCurves[0][0][6] = 330.0;
    filterCurves[0][1][6] = 0.560;
    filterCurves[0][0][7] = 335.0;
    filterCurves[0][1][7] = 0.673;
    filterCurves[0][0][8] = 340.0;
    filterCurves[0][1][8] = 0.772;
    filterCurves[0][0][9] = 345.0;
    filterCurves[0][1][9] = 0.841;
    filterCurves[0][0][10] = 350.0;
    filterCurves[0][1][10] = 0.905;
    filterCurves[0][0][11] = 355.0;
    filterCurves[0][1][11] = 0.943;
    filterCurves[0][0][12] = 360.0;
    filterCurves[0][1][12] = 0.981;
    filterCurves[0][0][13] = 365.0;
    filterCurves[0][1][13] = 0.993;
    filterCurves[0][0][14] = 370.0;
    filterCurves[0][1][14] = 1.000;
    filterCurves[0][0][15] = 375.0;
    filterCurves[0][1][15] = 0.989;
    filterCurves[0][0][16] = 380.0;
    filterCurves[0][1][16] = 0.916;
    filterCurves[0][0][17] = 385.0;
    filterCurves[0][1][17] = 0.804;
    filterCurves[0][0][18] = 390.0;
    filterCurves[0][1][18] = 0.625;
    filterCurves[0][0][19] = 395.0;
    filterCurves[0][1][19] = 0.423;
    filterCurves[0][0][20] = 400.0;
    filterCurves[0][1][20] = 0.238;
    filterCurves[0][0][21] = 405.0;
    filterCurves[0][1][21] = 0.114;
    filterCurves[0][0][22] = 410.0;
    filterCurves[0][1][22] = 0.051;
    filterCurves[0][0][23] = 415.0;
    filterCurves[0][1][23] = 0.019;
    filterCurves[0][0][24] = 420.0;
    filterCurves[0][1][24] = 0.000;
    #//BX
    filterCurves[1][0][0] = 360.0;
    filterCurves[1][1][0] = 0.000;
    filterCurves[1][0][1] = 370.0;
    filterCurves[1][1][1] = 0.026;
    filterCurves[1][0][2] = 380.0;
    filterCurves[1][1][2] = 0.120;
    filterCurves[1][0][3] = 390.0;
    filterCurves[1][1][3] = 0.523;
    filterCurves[1][0][4] = 400.0;
    filterCurves[1][1][4] = 0.875;
    filterCurves[1][0][5] = 410.0;
    filterCurves[1][1][5] = 0.956;
    filterCurves[1][0][6] = 420.0;
    filterCurves[1][1][6] = 1.000;
    filterCurves[1][0][7] = 430.0;
    filterCurves[1][1][7] = 0.998;
    filterCurves[1][0][8] = 440.0;
    filterCurves[1][1][8] = 0.972;
    filterCurves[1][0][9] = 450.0;
    filterCurves[1][1][9] = 0.901;
    filterCurves[1][0][10] = 460.0;
    filterCurves[1][1][10] = 0.793;
    filterCurves[1][0][11] = 470.0;
    filterCurves[1][1][11] = 0.694;
    filterCurves[1][0][12] = 480.0;
    filterCurves[1][1][12] = 0.587;
    filterCurves[1][0][13] = 490.0;
    filterCurves[1][1][13] = 0.470;
    filterCurves[1][0][14] = 500.0;
    filterCurves[1][1][14] = 0.362;
    filterCurves[1][0][15] = 510.0;
    filterCurves[1][1][15] = 0.263;
    filterCurves[1][0][16] = 520.0;
    filterCurves[1][1][16] = 0.169;
    filterCurves[1][0][17] = 530.0;
    filterCurves[1][1][17] = 0.107;
    filterCurves[1][0][18] = 540.0;
    filterCurves[1][1][18] = 0.049;
    filterCurves[1][0][19] = 550.0;
    filterCurves[1][1][19] = 0.010;
    filterCurves[1][0][20] = 560.0;
    filterCurves[1][1][20] = 0.000;
    filterCurves[1][0][21] = 560.0;
    filterCurves[1][1][21] = 0.000;
    filterCurves[1][0][22] = 560.0;
    filterCurves[1][1][22] = 0.000;
    filterCurves[1][0][23] = 560.0;
    filterCurves[1][1][23] = 0.000;
    filterCurves[1][0][24] = 560.0;
    filterCurves[1][1][24] = 0.000;
    #//B
    filterCurves[2][0][0] = 360.0;
    filterCurves[2][1][0] = 0.000;
    filterCurves[2][0][1] = 370.0;
    filterCurves[2][1][1] = 0.030;
    filterCurves[2][0][2] = 380.0;
    filterCurves[2][1][2] = 0.134;
    filterCurves[2][0][3] = 390.0;
    filterCurves[2][1][3] = 0.567;
    filterCurves[2][0][4] = 400.0;
    filterCurves[2][1][4] = 0.920;
    filterCurves[2][0][5] = 410.0;
    filterCurves[2][1][5] = 0.978;
    filterCurves[2][0][6] = 420.0;
    filterCurves[2][1][6] = 1.000;
    filterCurves[2][0][7] = 430.0;
    filterCurves[2][1][7] = 0.978;
    filterCurves[2][0][8] = 440.0;
    filterCurves[2][1][8] = 0.935;
    filterCurves[2][0][9] = 450.0;
    filterCurves[2][1][9] = 0.853;
    filterCurves[2][0][10] = 460.0;
    filterCurves[2][1][10] = 0.740;
    filterCurves[2][0][11] = 470.0;
    filterCurves[2][1][11] = 0.640;
    filterCurves[2][0][12] = 480.0;
    filterCurves[2][1][12] = 0.536;
    filterCurves[2][0][13] = 490.0;
    filterCurves[2][1][13] = 0.424;
    filterCurves[2][0][14] = 500.0;
    filterCurves[2][1][14] = 0.325;
    filterCurves[2][0][15] = 510.0;
    filterCurves[2][1][15] = 0.235;
    filterCurves[2][0][16] = 520.0;
    filterCurves[2][1][16] = 0.150;
    filterCurves[2][0][17] = 530.0;
    filterCurves[2][1][17] = 0.095;
    filterCurves[2][0][18] = 540.0;
    filterCurves[2][1][18] = 0.043;
    filterCurves[2][0][19] = 550.0;
    filterCurves[2][1][19] = 0.009;
    filterCurves[2][0][20] = 560.0;
    filterCurves[2][1][20] = 0.000;
    filterCurves[2][0][21] = 560.0;
    filterCurves[2][1][21] = 0.000;
    filterCurves[2][0][22] = 560.0;
    filterCurves[2][1][22] = 0.000;
    filterCurves[2][0][23] = 560.0;
    filterCurves[2][1][23] = 0.000;
    filterCurves[2][0][24] = 560.0;
    filterCurves[2][1][24] = 0.000;
    #//V
    filterCurves[3][0][0] = 470.0;
    filterCurves[3][1][0] = 0.000;
    filterCurves[3][0][1] = 480.0;
    filterCurves[3][1][1] = 0.030;
    filterCurves[3][0][2] = 490.0;
    filterCurves[3][1][2] = 0.163;
    filterCurves[3][0][3] = 500.0;
    filterCurves[3][1][3] = 0.458;
    filterCurves[3][0][4] = 510.0;
    filterCurves[3][1][4] = 0.780;
    filterCurves[3][0][5] = 520.0;
    filterCurves[3][1][5] = 0.967;
    filterCurves[3][0][6] = 530.0;
    filterCurves[3][1][6] = 1.000;
    filterCurves[3][0][7] = 540.0;
    filterCurves[3][1][7] = 0.973;
    filterCurves[3][0][8] = 550.0;
    filterCurves[3][1][8] = 0.898;
    filterCurves[3][0][9] = 560.0;
    filterCurves[3][1][9] = 0.792;
    filterCurves[3][0][10] = 570.0;
    filterCurves[3][1][10] = 0.684;
    filterCurves[3][0][11] = 580.0;
    filterCurves[3][1][11] = 0.574;
    filterCurves[3][0][12] = 590.0;
    filterCurves[3][1][12] = 0.461;
    filterCurves[3][0][13] = 600.0;
    filterCurves[3][1][13] = 0.359;
    filterCurves[3][0][14] = 610.0;
    filterCurves[3][1][14] = 0.270;
    filterCurves[3][0][15] = 620.0;
    filterCurves[3][1][15] = 0.197;
    filterCurves[3][0][16] = 630.0;
    filterCurves[3][1][16] = 0.135;
    filterCurves[3][0][17] = 640.0;
    filterCurves[3][1][17] = 0.081;
    filterCurves[3][0][18] = 650.0;
    filterCurves[3][1][18] = 0.045;
    filterCurves[3][0][19] = 660.0;
    filterCurves[3][1][19] = 0.025;
    filterCurves[3][0][20] = 670.0;
    filterCurves[3][1][20] = 0.017;
    filterCurves[3][0][21] = 680.0;
    filterCurves[3][1][21] = 0.013;
    filterCurves[3][0][22] = 690.0;
    filterCurves[3][1][22] = 0.009;
    filterCurves[3][0][23] = 700.0;
    filterCurves[3][1][23] = 0.000;
    filterCurves[3][0][24] = 700.0;
    filterCurves[3][1][24] = 0.000;
    #//R
    filterCurves[4][0][0] = 550.0;
    filterCurves[4][1][0] = 0.00;
    filterCurves[4][0][1] = 560.0;
    filterCurves[4][1][1] = 0.23;
    filterCurves[4][0][2] = 570.0;
    filterCurves[4][1][2] = 0.74;
    filterCurves[4][0][3] = 580.0;
    filterCurves[4][1][3] = 0.91;
    filterCurves[4][0][4] = 590.0;
    filterCurves[4][1][4] = 0.98;
    filterCurves[4][0][5] = 600.0;
    filterCurves[4][1][5] = 1.00;
    filterCurves[4][0][6] = 610.0;
    filterCurves[4][1][6] = 0.98;
    filterCurves[4][0][7] = 620.0;
    filterCurves[4][1][7] = 0.96;
    filterCurves[4][0][8] = 630.0;
    filterCurves[4][1][8] = 0.93;
    filterCurves[4][0][9] = 640.0;
    filterCurves[4][1][9] = 0.90;
    filterCurves[4][0][10] = 650.0;
    filterCurves[4][1][10] = 0.86;
    filterCurves[4][0][11] = 660.0;
    filterCurves[4][1][11] = 0.81;
    filterCurves[4][0][12] = 670.0;
    filterCurves[4][1][12] = 0.78;
    filterCurves[4][0][13] = 680.0;
    filterCurves[4][1][13] = 0.72;
    filterCurves[4][0][14] = 690.0;
    filterCurves[4][1][14] = 0.67;
    filterCurves[4][0][15] = 700.0;
    filterCurves[4][1][15] = 0.61;
    filterCurves[4][0][16] = 710.0;
    filterCurves[4][1][16] = 0.56;
    filterCurves[4][0][17] = 720.0;
    filterCurves[4][1][17] = 0.51;
    filterCurves[4][0][18] = 730.0;
    filterCurves[4][1][18] = 0.46;
    filterCurves[4][0][19] = 740.0;
    filterCurves[4][1][19] = 0.40;
    filterCurves[4][0][20] = 750.0;
    filterCurves[4][1][20] = 0.35;
    filterCurves[4][0][21] = 800.0;
    filterCurves[4][1][21] = 0.14;
    filterCurves[4][0][22] = 850.0;
    filterCurves[4][1][22] = 0.03;
    filterCurves[4][0][23] = 900.0;
    filterCurves[4][1][23] = 0.00;
    filterCurves[4][0][24] = 900.0;
    filterCurves[4][1][24] = 0.000;

    #//I
    filterCurves[5][0][0] = 700.0;
    filterCurves[5][1][0] = 0.000;
    filterCurves[5][0][1] = 710.0;
    filterCurves[5][1][1] = 0.024;
    filterCurves[5][0][2] = 720.0;
    filterCurves[5][1][2] = 0.232;
    filterCurves[5][0][3] = 730.0;
    filterCurves[5][1][3] = 0.555;
    filterCurves[5][0][4] = 740.0;
    filterCurves[5][1][4] = 0.785;
    filterCurves[5][0][5] = 750.0;
    filterCurves[5][1][5] = 0.910;
    filterCurves[5][0][6] = 760.0;
    filterCurves[5][1][6] = 0.965;
    filterCurves[5][0][7] = 770.0;
    filterCurves[5][1][7] = 0.985;
    filterCurves[5][0][8] = 780.0;
    filterCurves[5][1][8] = 0.990;
    filterCurves[5][0][9] = 790.0;
    filterCurves[5][1][9] = 0.995;
    filterCurves[5][0][10] = 800.0;
    filterCurves[5][1][10] = 1.000;
    filterCurves[5][0][11] = 810.0;
    filterCurves[5][1][11] = 1.000;
    filterCurves[5][0][12] = 820.0;
    filterCurves[5][1][12] = 0.990;
    filterCurves[5][0][13] = 830.0;
    filterCurves[5][1][13] = 0.980;
    filterCurves[5][0][14] = 840.0;
    filterCurves[5][1][14] = 0.950;
    filterCurves[5][0][15] = 850.0;
    filterCurves[5][1][15] = 0.910;
    filterCurves[5][0][16] = 860.0;
    filterCurves[5][1][16] = 0.860;
    filterCurves[5][0][17] = 870.0;
    filterCurves[5][1][17] = 0.750;
    filterCurves[5][0][18] = 880.0;
    filterCurves[5][1][18] = 0.560;
    filterCurves[5][0][19] = 890.0;
    filterCurves[5][1][19] = 0.330;
    filterCurves[5][0][20] = 900.0;
    filterCurves[5][1][20] = 0.150;
    filterCurves[5][0][21] = 910.0;
    filterCurves[5][1][21] = 0.030;
    filterCurves[5][0][22] = 920.0;
    filterCurves[5][1][22] = 0.000;
    filterCurves[5][0][23] = 920.0;
    filterCurves[5][1][23] = 0.000;
    filterCurves[5][0][24] = 920.0;
    filterCurves[5][1][24] = 0.000;
#//HJK from Johnson, H.L, 1965, ApJ 141, 923
#//http://ulisse.pd.astro.it/Astro/ADPS/Systems/Sys_033/index_033.html    
#//H lburns /06
    filterCurves[6][0][0] = 1460;
    filterCurves[6][1][0] = 0.000;
    filterCurves[6][0][1] = 1480;
    filterCurves[6][1][1] = 0.150;
    filterCurves[6][0][2] = 1500;
    filterCurves[6][1][2] = 0.440;
    filterCurves[6][0][3] = 1520;
    filterCurves[6][1][3] = 0.860;
    filterCurves[6][0][4] = 1540;
    filterCurves[6][1][4] = 0.940;
    filterCurves[6][0][5] = 1550;
    filterCurves[6][1][5] = 0.960;
    filterCurves[6][0][6] = 1560;
    filterCurves[6][1][6] = 0.980;
    filterCurves[6][0][7] = 1580;
    filterCurves[6][1][7] = 0.950;
    filterCurves[6][0][8] = 1600;
    filterCurves[6][1][8] = 0.990;
    filterCurves[6][0][9] = 1610;
    filterCurves[6][1][9] = 0.990;
    filterCurves[6][0][10] = 1620;
    filterCurves[6][1][10] = 0.990;
    filterCurves[6][0][11] = 1640;
    filterCurves[6][1][11] = 0.990;
    filterCurves[6][0][12] = 1660;
    filterCurves[6][1][12] = 0.990;
    filterCurves[6][0][13] = 1670;
    filterCurves[6][1][13] = 0.990;
    filterCurves[6][0][14] = 1680;
    filterCurves[6][1][14] = 0.990;
    filterCurves[6][0][15] = 1690;
    filterCurves[6][1][15] = 0.990;
    filterCurves[6][0][16] = 1700;
    filterCurves[6][1][16] = 0.990;
    filterCurves[6][0][17] = 1710;
    filterCurves[6][1][17] = 0.970;
    filterCurves[6][0][18] = 1720;
    filterCurves[6][1][18] = 0.950;
    filterCurves[6][0][19] = 1740;
    filterCurves[6][1][19] = 0.870;
    filterCurves[6][0][20] = 1760;
    filterCurves[6][1][20] = 0.840;
    filterCurves[6][0][21] = 1780;
    filterCurves[6][1][21] = 0.710;
    filterCurves[6][0][22] = 1800;
    filterCurves[6][1][22] = 0.520;
    filterCurves[6][0][23] = 1820;
    filterCurves[6][1][23] = 0.020;
    filterCurves[6][0][24] = 1840;
    filterCurves[6][1][24] = 0.000;
#//J lburns /06
    filterCurves[7][0][0] = 1040;
    filterCurves[7][1][0] = 0.000;
    filterCurves[7][0][1] = 1060;
    filterCurves[7][1][1] = 0.020;
    filterCurves[7][0][2] = 1080;
    filterCurves[7][1][2] = 0.110;
    filterCurves[7][0][3] = 1100;
    filterCurves[7][1][3] = 0.420;
    filterCurves[7][0][4] = 1120;
    filterCurves[7][1][4] = 0.320;
    filterCurves[7][0][5] = 1140;
    filterCurves[7][1][5] = 0.470;
    filterCurves[7][0][6] = 1160;
    filterCurves[7][1][6] = 0.630;
    filterCurves[7][0][7] = 1180;
    filterCurves[7][1][7] = 0.730;
    filterCurves[7][0][8] = 1190;
    filterCurves[7][1][8] = 0.750;
    filterCurves[7][0][9] = 1200;
    filterCurves[7][1][9] = 0.770;
    filterCurves[7][0][10] = 1210;
    filterCurves[7][1][10] = 0.790;
    filterCurves[7][0][11] = 1220;
    filterCurves[7][1][11] = 0.810;
    filterCurves[7][0][12] = 1230;
    filterCurves[7][1][12] = 0.820;
    filterCurves[7][0][13] = 1240;
    filterCurves[7][1][13] = 0.830;
    filterCurves[7][0][14] = 1250;
    filterCurves[7][1][14] = 0.850;
    filterCurves[7][0][15] = 1260;
    filterCurves[7][1][15] = 0.880;
    filterCurves[7][0][16] = 1280;
    filterCurves[7][1][16] = 0.940;
    filterCurves[7][0][17] = 1300;
    filterCurves[7][1][17] = 0.910;
    filterCurves[7][0][18] = 1320;
    filterCurves[7][1][18] = 0.790;
    filterCurves[7][0][19] = 1340;
    filterCurves[7][1][19] = 0.680;
    filterCurves[7][0][20] = 1360;
    filterCurves[7][1][20] = 0.040;
    filterCurves[7][0][21] = 1380;
    filterCurves[7][1][21] = 0.110;
    filterCurves[7][0][22] = 1400;
    filterCurves[7][1][22] = 0.070;
    filterCurves[7][0][23] = 1420;
    filterCurves[7][1][23] = 0.030;
    filterCurves[7][0][24] = 1440;
    filterCurves[7][1][24] = 0.000;
#//K lburns /06  
    filterCurves[8][0][0] = 1940;
    filterCurves[8][1][0] = 0.000;
    filterCurves[8][0][1] = 1960;
    filterCurves[8][1][1] = 0.120;
    filterCurves[8][0][2] = 1980;
    filterCurves[8][1][2] = 0.200;
    filterCurves[8][0][3] = 2000;
    filterCurves[8][1][3] = 0.300;
    filterCurves[8][0][4] = 2020;
    filterCurves[8][1][4] = 0.550;
    filterCurves[8][0][5] = 2040;
    filterCurves[8][1][5] = 0.740;
    filterCurves[8][0][6] = 2060;
    filterCurves[8][1][6] = 0.550;
    filterCurves[8][0][7] = 2080;
    filterCurves[8][1][7] = 0.770;
    filterCurves[8][0][8] = 2100;
    filterCurves[8][1][8] = 0.850;
    filterCurves[8][0][9] = 2120;
    filterCurves[8][1][9] = 0.900;
    filterCurves[8][0][10] = 2140;
    filterCurves[8][1][10] = 0.940;
    filterCurves[8][0][11] = 2160;
    filterCurves[8][1][11] = 0.940;
    filterCurves[8][0][12] = 2180;
    filterCurves[8][1][12] = 0.950;
    filterCurves[8][0][13] = 2200;
    filterCurves[8][1][13] = 0.940;
    filterCurves[8][0][14] = 2220;
    filterCurves[8][1][14] = 0.960;
    filterCurves[8][0][15] = 2240;
    filterCurves[8][1][15] = 0.980;
    filterCurves[8][0][16] = 2260;
    filterCurves[8][1][16] = 0.970;
    filterCurves[8][0][17] = 2280;
    filterCurves[8][1][17] = 0.960;
    filterCurves[8][0][18] = 2300;
    filterCurves[8][1][18] = 0.910;
    filterCurves[8][0][19] = 2320;
    filterCurves[8][1][19] = 0.880;
    filterCurves[8][0][20] = 2340;
    filterCurves[8][1][20] = 0.840;
    filterCurves[8][0][21] = 2380;
    filterCurves[8][1][21] = 0.750;
    filterCurves[8][0][22] = 2400;
    filterCurves[8][1][22] = 0.640;
    filterCurves[8][0][23] = 2440;
    filterCurves[8][1][23] = 0.010;
    filterCurves[8][0][24] = 2480;
    filterCurves[8][1][24] = 0.000;
    
    #//
    #//Check that we set up the array corectly:
    #//    for (var ib = 0; ib < numBands; ib++) {
    #//    var ib = 0;
    #//    for (var il = 0; il < numLambdaFilt; il++) {
    #//        console.log("ib: " + ib + " il: " + il + " filterCurves[ib][0][il]: " + filterCurves[0][0][il]);
    #//       console.log("ib: " + ib + " il: " + il + " filterCurves[ib][1][il]: " + filterCurves[0][1][il]);
    #//   }
    #//    }

    for ib in range(numBands):
        #//wavelength loop is over photometric filter data wavelengths
        for il in range(numLambdaFilt):
            filterCurves[ib][0][il] = filterCurves[ib][0][il] * nm2cm #// nm to cm
        #}
    #}


    return filterCurves
#}; //filterSet


#/* In case it's ever needed again...
#//General convolution method
#// ***** Function to be convolved and kernel function are expected to *already* be 
#// interpolated onto same abssica grid!
#//
def convol(x, yFunction, kernel):

    ySize = len(yFunction)
    kernelSize = len(kernel)
    halfKernelSize = math.ceil(kernelSize / 2)
  
    yFuncConv = [0.0 for i in range(ySize)]
    #var deltaX;

    #//First kernelSize/2 elements of yFunction cannot be convolved
    for i in range(halfKernelSize):
        yFuncConv[i] = yFunction[i]
        #//console.log("Part 1: i " + i + " yFuncConv[i] " + yFuncConv[i]);
        
    #//Convolution:
    #//We are effectively integrating in pixel space, not physical wavelength space, so deltaX is always unity - ??
    #// Conserves power if kernel area-normalized - ??
    offset = 0 #//initialization
    for i in range(halfKernelSize, ySize - (halfKernelSize)):
        accum = 0 #//accumulator
        for j in range(kernelSize): 
            #//console.log("Part 2: i " + i + " j " + j + " offset " + offset);
            #//deltaX = x[j] - x[j-1]; 
            #//console.log("x[j] " + x[j] + " x[j-1] " + x[j-1] + " deltaX " + deltaX 
            #// + " yFunction[j+offset] " + yFunction[j+offset] + " kernel[j] " + kernel[j]);
            accum = accum + ( (kernel[j] * yFunction[j+offset]) ) #//* deltaX ); 
        #}  //inner loop, j	
        yFuncConv[i] = accum
        #//console.log("yFuncConv[i] " + yFuncConv[i]);
        offset+=1
    #} //outer loop, i
    #//Last kernelSize/2 elements of yFunction cannot be convolved
    for i in range(ySize - halfKernelSize - 1, ySize):
        yFuncConv[i] = yFunction[i]
    #}

    return yFuncConv 

#}; //end method convol

"""
*/   
/**
 *
 * THIS VERSION works with spectrum synthesis output returned from server
 * in GrayStarServer
 *"""
 
def eqWidthSynth(flux, linePoints): 
    #//, fluxCont) {

    """* It will try to return the equivalenth width of EVERYTHING in the synthesis region
 * as one value!  Isolate the synthesis region to a single line to a clean result
 * for that line!
 *
 * Compute the equivalent width of the Voigt line in pm - picometers NOTE: The
 * input parameter 'flux' should be a 2 x (numPoints+1) array where the
 * numPoints+1st value is the line centre monochromatic Continuum flux
 */"""
 
    logE = math.log10(math.e) #// for debug output
    Wlambda = 0.0 #// Equivalent width in pm - picometers
    numPoints = len(linePoints)
    #//console.log("numPoints " + numPoints);
    #var delta, logDelta, term, integ, integ2, logInteg, lastInteg, lastTerm, term2;

    #//Spectrum now continuum rectified before eqWidth called

    #//Trapezoid rule:
    #// First integrand:

    lastInteg = 1.0 - flux[0][0]
    
    lastTerm = lastInteg #//initialization

    for il in range(1, numPoints-1):

        delta = linePoints[il] - linePoints[il - 1]
        delta = delta * 1.0E+7  #// cm to nm - W_lambda in pm
        logDelta = math.log(delta)

        integ = 1.0 - flux[0][il]

        #//Extended trapezoid rule:
        integ2 = 0.5 * (lastInteg + integ)
        #//logInteg = math.log(integ2)
        #//term = Math.exp(logInteg + logDelta);
        term = integ2 * delta
        #//console.log("linePoints[il] " + linePoints[il] + " flux[0][il] " + flux[0][il]
        #//  + " integ " + integ + " term " + term);

        #//Wlambda = Wlambda + (term * delta);
        Wlambda = Wlambda + term

        lastInteg = integ

        #//System.out.println("EqWidth: Wlambda: " + Wlambda);
    #}

    #// Convert area in nm to pm - picometers
    Wlambda = Wlambda * 1.0E3

    return Wlambda

#};


 

def fourier(numThetas, cosTheta, filtIntens):
    
    """//Discrete cosine and sine Fourier transform of input narrow-band intensity profile"""
    
    # //
    #// We will interpret theta/(theta/2) with respect to the local surface normal of the star to
    #// be the spatial domain "x" coordinate - this is INDEPENDENT of the distance to, and linear
    #// radius of, the star! :-)

    pi = math.pi  #//a handy enough wee quantity
    halfPi = pi / 2.0

    #//number of sample points in full intensity profile I(theta), theta = -pi/2 to pi/2 RAD:
    numX0 = 2 * numThetas - 1

    #//We have as input the itnesity half-profile I(cos(theta)), cos(theta) = 1 to 0
    #//create the doubled root-intensity profile sqrt(I(theta/halfPi)), theta/halfPi = -1 to 1
    #//this approach assumes the real (cosine) and imaginary (sine) components are in phase
    rootIntens2 = [0.0 for i in range(numX0)]
    x0 = [0.0 for i in range(numX0)]

    #var normIntens;
    #//negative x domain of doubled profile:
    j = 0
    for i in range(numThetas-1, 0, -1):
        x0[j] = -1.0*math.acos(cosTheta[1][i]) / halfPi
        normIntens = filtIntens[i] / filtIntens[0] #//normalize
        rootIntens2[j] = math.sqrt(normIntens)
        #//console.log("i " + i + " cosTheta " + cosTheta[1][i] + " filtIntens " + filtIntens[i] + " normIntens " + normIntens
        #//  + " j " + j + " x0 " + x0[j] + " rootIntens2 " + rootIntens2[j] );
        j+=1
    #}
    #//positive x domain of doubled profile:
    for i in range(numThetas, numX0):
        j = i - (numThetas-1)
        x0[i] = math.acos(cosTheta[1][j]) / halfPi
        normIntens = filtIntens[j] / filtIntens[0] #//normalize
        #//rootIntens2[i] = Math.sqrt(normIntens);
        rootIntens2[i] = normIntens
        #//console.log("j " + j + " cosTheta " + cosTheta[1][j] + " filtIntens " + filtIntens[j] + " normIntens " + normIntens
        #//  + " i " + i + " x0 " + x0[i] + " rootIntens2 " + rootIntens2[i] );
    #}

    #//create the uniformly sampled spatial domain ("x") and the complementary
    #//spatial frequecy domain "k" domain
    #//
    #//We're interpreting theta/halfPi with respect to local surface normal at surface
    #//of star as the spatial domain, "x"
    minX = -2.0
    maxX = 1.0
    numX = 100  #//(is also "numK" - ??)
    deltaX = (maxX - minX) / numX

    #//Complentary limits in "k" domain; k = 2pi/lambda (radians)
    #//  - lowest k value corresponds to one half spatial wavelength (lambda) = 2 (ie. 1.0 - (-1.0)):
    maxLambda = 2.0 * 2.0
    #// stupid?? var minK = 2.0 * pi / maxLambda;  //(I know, I know, but let's keep this easy for the human reader)
    #//  - highest k value has to do with number of points sampling x:  Try Nyquist sampling rate of
    #//     two x points per lambda
    minLambda = 8.0 * 2.0 * deltaX
    maxK = 2.0 * pi / minLambda #//"right-going" waves
    minK = -1.0 * maxK  #//"left-going" waves
    deltaK = (maxK - minK) / numX
    #// console.log("maxK " + maxK + " minK " + minK + " deltaK " + deltaK);


    x = [0.0 for i in range(numX)]
    k = [0.0 for i in range(numX)]

    #var ii;
    for i in range(numX):
        ii = 1.0 * i
        x[i] = minX + ii*deltaX
        k[i] = minK + ii*deltaK
        #// console.log("i " + i + " x " + x[i] + " k " + k[i]);
  

    #//Interpolate the rootIntens2(theta/halfpi) signal onto uniform spatial sampling:
    #//doesn't work: var rootIntens3 = interpolV(rootIntens2, x0, x);
    rootIntens3 = [0.0 for i in range(numX)]

    for i in range(numX):
        rootIntens3[i] = ToolBox.interpol(x0, rootIntens2, x[i])
        #//console.log("i " + i + " x " + x[i] + " rootIntens3 " + rootIntens3[i]);
  

    #//returned variable ft:
    #//  Row 0: wavenumber, spatial frequency, k (radians)
    #//  Row 1: cosine transform (real component)
    #//  Row 2: sine transform (imaginary component
    ft = [ [ 0.0 for i in range(numX-1) ] for j in range(3) ]

    #var argument, rootFt;
    #//numXFloat = 1.0 * numX;
    #//Outer loop is over the elements of vector holding the power at each frequency "k"
    for ik in range(numX-1):
        #//initialize ft
        ft[0][ik] = k[ik]
        rootFtCos = 0.0
        rootFtSin = 0.0
        ft[1][ik] = 0.0
        ft[2][ik] = 0.0
        #//Inner llop is cumulative summation over spatial positions "x" - the Fourier cosine and sine series
        for ix in range(numX-1):
            #//ixFloat = 1.0 * ix;
            argument = -1.0 * k[ik] * x[ix]
            #//console.log("ik " + ik + " ix " + ix + " argument " + argument + " x " + x[ix] + " rootIntens3 " + rootIntens3[ix]);
            #// cosine series:
            rootFtCos = rootFtCos + rootIntens3[ix] * math.cos(argument)
            #// sine series:
            rootFtSin = rootFtSin + rootIntens3[ix] * math.sin(argument);
        #} //ix loop
        ft[1][ik] = rootFtCos #// * rootFtCos; //Power
        ft[2][ik] = rootFtSin #// * rootFtSin;
        #//console.log("ik " + ik + " k " + k[ik] + " ft[1] " + ft[1][ik]);
    #} //ik loop

    return ft

#}; //end method fourier
back to top