A collection of remixable spectrum analysis tools.
The Spectrum Viewer pages can be served as static pages from gh-pages, or anywhere you like. server.js provides a minimal node server that will do the job; dependencies and setup are at the top of that file.
In order to point your Spectrum Viewer at the appropriate spectra, a couple of parameters need to be set in scripts/spectrumViewer.js:
- dataStore.spectrumServershould be the url with port where raw spectra are served according to the API described below in 'Data Feeds'.
- The topGroupsobject describes the plot selection and navigation; set names and ids of groups of plots as desired, and make sure theitemskey contains an array listing the exact names of plots to include in the corresponding subgroup.
In order to point your 2D Spectrum Viewer at the appropriate spectra, a couple of parameters need to be set in scripts/2dSpectrumViewer.js:
- dataStore.spectrumServershould be the url with port where raw spectra are served according to the API described below in 'Data Feeds'.
- dataStore.ODBhostshould be the URL and host of the MIDAS experiment to write region of interest parameterizations to.
- The topGroupsobject describes the plot selection and navigation; set names and ids of groups of plots as desired, and make sure theitemskey contains an array listing the exact names of plots to include in the corresponding subgroup.
In order to point your Rate Monitor at the appropriate spectra, a couple of parameters need to be set in scripts/rateMonitor.js:
- dataStore.spectrumServershould be the url with port where raw spectra are served according to the API described below in 'Data Feeds'.
- dataStore.ODBrequestsshould be an array of ODB requests to perform on refresh; see specifications here.
In order to point your Gain Matcher at the appropriate spectra, a couple of parameters need to be set in scripts/gainMatcher.js:
- dataStore.spectrumServershould be the url with port where raw spectra are served according to the API described below in 'Data Feeds'.
- dataStore.ODBhost: host and port of MIDAS ODB experiment.
- dataStore.ODBrequestsshould be an array of ODB requests to perform before writing calibration constants to the MSC table; see request specifications here.
Contributions are very welcome! If you have an idea, question or comment, please open an issue. If you would like to make a change to this project, please follow these steps:
- start by opening an issue or empty PR to discuss your ideas
- please limit individual PRs to less than 500 lines (Why? See figure 1 here).
- please encapsulate all new behavior wherever possible in functions of 50 lines or less each.
All these tools rely on being able to query spectrum data from a server providing JSON reponses. The expected API is as follows:
?cmd=getSpectrumList
Returns an array of all available spectrum names in the format:
getSpectrumList({'spectrumlist':['firstSpectrum', 'nextSpectrum', ..., 'lastSpectrum']})
?cmd=callspechandler&spectrum0=firstSpecName&spectrum1=nextSpecName&...
where the nth SpecName matches a string in the array returned by getSpectrumList, returns a JSON object that contains an element of the form name : array of bin contents for each 1D spectrum requested:
{'firstSpecName' : [0, 3, 2, 7, ...], 'nextSpecName' : [1,5,2,9, ...], ....}
or, for 2D spectra:
{'2dSpectrumName': [rowLength, z00, z10, ..., z01, z11,.... ]}
where rowLength is the number of bins in x in one row, and the z values of all bins are packed counting first along x bins, then along y bins.
If the named spectrum is not a valid spectrum, the return object above should contain an entry 'specName':null.
All these projects are built on a collection of objects and HTML snippets encapsulated and imported via HTML Templates and HTML Imports. All of these objects exchange information via a collection of custom events, as illustrated below, as well as through a global dataStore object. Layout is ala Bootstrap, and client-side templating is built on mustache.js.
All these apps share some common design features:
- 
Pageload timing looks like: - load global dataStoreobject (see below for description)
- sketch out dom using wrapper <div>s
- main scripting routine proceeds once HTMLImportsLoadedfires, indicating that the HTML templates and accompanying scripts are ready to go.
- Most major page elements consist of an HTML template, accompanied by a JavaScript function that makes an object to namespace and manage the behavior of that element. Once HTMLImportsLoadedfires, these functions are called as constructors and their.setup()routine is invoked to set them up.
 
- load global 
- 
Each app uses a global dataStoreobject to namespace global variables.dataStoreis initialized by a functionsetupDataStore, called immediately in the head. This is the appropriate place to add global variables. Documentation ondataStorekeys is found inline with their declaration; note that a number of template objects draw their configurations fromdataStore. Most notably,dataStorecontains a keyplots, an array of strings which label thegammaspectrum.jsplotting objects available. Those plotting objects in turn live on an objectdataStore.viewers, for easy global access by all elements; the majority of inter-component communication not encapsulated by a custom event is through reads and writes to these viewer objects.
- 
Network requests are done via a series of Promises and callbacks. When adding new network requests, it is important to respect this pattern if you want responses to be dealt with at the correct time. The pattern is: - All analyzer requests (per the spec above) are sent in parallel. Spectra are added to the appropriate plots and logged in the dataStoreas they arrive.
- On receipt of all analyzer info, all ODB requests are sent in parallel, and processed by the function named in the request's callbackquery string parameter.
- After completion of all ODB requests, a global function fetchCallbackis run, if it exists.
 This structure is found in templates/plotControl/plotControl.js, inrefreshAll(), the central function responsible for orchestrating data refreshes over the network. In order to add an analyzer request, the appropriate URL must be added to thequeriesarray found there; in order to add an ODB request, modify the string found indataStore.ODBrequests[0], per the ODB's AJAX spec. In order to take action on reciept of ODB information, modifyparseODB(); in order to take action immediately after all data has been refreshed, modifyfetchCallback().
- All analyzer requests (per the spec above) are sent in parallel. Spectra are added to the appropriate plots and logged in the 
The Spectrum Viewer elements communicate amongst themselves as above; nodes represent objects, and edges represent custom events. Basic behavior:
- clicking on a spectrum name in plotListfires arequestPlotevent, which tellsplotControlto queue up that histogram for plotting in all cells currently active, and subsequently dispatches anaddPlotRowevent to add a row to the appropriate tables inauxPlotControl.
- plotControlgoverns data refresh as described in section 0.1.
- plotGridfires- newCelland- deleteCellevents at the control elements when a new plot is created or destroyed, to keep controls up to date; also,- attachCellevents are sent to- plotControlto attach or unnattach the UI to each individual cell.
- fetchCallback()refreshes all plots once data has arrived.
The 2D Spectrum Viewer elements communicate amongst themselves as above; nodes represent objects, and edges represent custom events. Basic behavior:
- clicking on a spectrum name in plotListfires arequestPlotevent, which tellsplotControlto plot the selected histogram in theheatmapelement.
- fetchCallback()refreshes all plots once data has arrived.
Note that this viewer relies on the heatmap package, a high-performance library for quickly drawing histograms with large numbers of bins.
The Rate Monitor elements communicate amongst themselves as above; nodes represent objects, and edges represent custom events. Basic behavior:
- As usual, plotControlis responsible for periodically refreshing spectrum and ODB data, in the same manner as above.
- fetchCallbacktriggers a series of calculations after data is refreshed:- Linear background fits are performed around all the active gamma windows per the background subtraction options selected in rateControl, and those backgrounds are subtracted from their corresponding windows.
- A rate is calculated by comparing the number of counts in each window, to the number of counts in the corresponding window (after background subtraction) the last time the spectrum was fetched; time between polls is estimated as the interval between receipt times of each poll of the spectrum at the browser.
- The dygraph tracking rates and scalers as well as annotations is updated, per the window defined by rateSlider.
 
- Linear background fits are performed around all the active gamma windows per the background subtraction options selected in 
- rateControlcan send events to the dygraph to toggle series visibility (- setDyVisible) and update its data (- updateDyData).
- plotGridsends its- newCelland- attachCellevents to- plotControlexactly once in this case, to set up on page load.
The Gain Matcher elements communicate amongst themselves as above; nodes represent objects, and edges represent custom events. Basic behavior:
- Data for all 64 GRIFFIN crystals is fetched exactly once, on pageload in the same manner as above by plotControl.
- Upon initiating the gain matching procedure, gainMatchReportestimates the bins where the peaks requested will be found, fits a gaussian plus linear background to the tallest peak found there, and writes the result to its table. Upon completion, afitAllCompleteevent is sent toplotListLiteto set it to display the first plot in the series, and anupdateDyDataevent is sent tographto refresh the plot of peak widths as a function of detector.



