CROW is a single-page application to visualize birds detected by weather radars. It visualizes vpts
data from a public repository directly in the browser and can thus be hosted on a static file server. As it only visualizes data from one radar at a time, it is easily scalable.
CROW is implemented in TypeScript and makes use of the Vue CLI toolkit, D3 and BootstrapVue.
CROW was jointly developed by the Research Institute for Nature and Forest (INBO) and the Royal Meteorological Institute of Belgium (RMI) in collaboration with the Royal Belgian Institute for Natural Sciences (RBINS), with financial support from the Belgian Science Policy Office (BelSPO valorisation project CROW
).
The application can be cited as:
Noé N, Reyniers M, Van Hoey S, Desmet P (2021) CROW - Online tool to visualize birds detected by weather radars. http://doi.org/10.5281/zenodo.4629449
Install with:
npm install
Run in development with:
npm run serve
Run unit tests with:
npm run test:unit
Build with:
npm run build
There is no need to build manually, as this is done automatically by GitHub Actions for every commit pushed to the main
branch. It will serve the site at https://inbo.github.io/crow/.
If the app is to be deployed under a URL prefix (for example https://inbo.github.io/crow
), create a env.local
file in the source tree root with content: URL_PREFIX = "/crow/"
.
CROW is implemented as a hierarchy of Vue.js components that can be found in the src/components
directory. Understanding the role and interface of each component in the tree allows understanding the data flow:
App
├── NavBar
│ └── LanguageSelector
├── Home
│ ├── Introduction
│ ├── SiteSelector
│ │ └── SiteSelectorMap
│ ├── VPIChart
│ │ └── DailyLines
│ ├── VPChart
│ │ ├── DailyLines
│ │ └── ColorLegend
│ ├── TimelineChart
│ └── Partners
└── PageFooter
Note: Vuex is used as a centralized data store since many components act on the same data, and therefore share a common state. Since it was only introduced quite late during CROW's development, it is not yet used for all data that could be considered shared state (those live mainly in the Home
component). Two Vuex data stores with explicit names are currently implemented: ConfigStore
and UserChoicesStore
.
CROW's configuration and initial selection (form state) is kept in src/config.ts
.
The most important components of CROW are:
App
is the root component and implements a few global features such as URL management and page layout (toolbar, footer, ...) but is not very interesting in terms of data flow. It uses Vue Router to extract parameters from the URL and pass them to the Home
component.
Home
is the largest component and in charge of:
- displaying the main form (site, date, interval, ...)
- loading, transforming and passing data to other components in charge of the visualizations. That happens each time the form changes and at inital page load. This process is described in more detail below.
The source data are time series of vertical profiles (vpts
)generated by vol2bird. These are stored on a public file server. The text files contain fixed-width separated values of variables related to the presence of birds (density, speed, reflectivity, ...), grouped by date (first column), time (second column) and height (third column). The structure in the source data files is therefore a flat table. See for example the data file for the radar at site Helchteren on November 1st, 2020.
However for performance reasons, the Home
compoment holds these data in a radarVtps
variable organized as a tree of objects:
radarVtps (Object)
├── 1604185200000 (Object - timestamp)
│ ├── heightData (Object - vertical profile of birds for this timestamp, per altitude)
│ │ ├── 0 (Object)
│ │ │ ├── dd: NaN
│ │ │ ├── dens: 0
│ │ │ └── ...
│ │ ├── 1000 (Object)
│ │ │ ├── dd: 3.24
│ │ │ └── dens: 1.01
│ │ └── ...
│ └── sunAltitude: -53.03 (Number: Sun altitude at the radar site for at this timestamp )
└── 1604185500000 (Object - timestamp)
└── ...
- the
radarVtps
object is initialized according to the selected time range and radar. Sun altitudes at site are also computed and set. - data file(s) are loaded via AJAX, and their content is used to populate
radarVtps
(more specifically, the various properties in each timestamp -> heightdata entries). - these data are transformed via computed properties and passed to the child components (that are in charge of the visulization):
VPChart
receives a flattened version ofradarVtps
(radarVtpsAsArray
, similar to the structure of the initial data files)VPIChart
receives vertically integrated profiles (seeintegrateProfile
function inhelpers.ts
).TimelineChart
receives a simple array with the sun altitude for each shown time period.
It makes use of a few more child components for modularity reasons: SiteSelector
, Introduction
, ...
VPIChart
visualizes the integrated values (MTR, VID, ...) with D3. It uses the DailyLines
component to show vertical lines on the chart each day at midnight.
VPChart
visualizes the raw VTPS data as a heatmap (bird density in function of the time and altitude) with D3. It uses the DailyLines
component to show vertical lines on the chart each day at midnight and ColorLegend
to show a legend for the 3 availables colour scales.