A Shiny application developed by the Industrial Hygiene team of the Department
of Environmental and Occupational Health at the
École de Santé Publique of the
Université de Montréal (EPSUM) in collaboration
with Ununoctium (Jean-Mathieu Potvin).
Ununoctium designed the user interface (for versions
greater than or equal to 4.0.0) and currently maintains the web application
(but not scripts stored in scripts/).
This tool interprets a dataset of exposure measurements (including non detects) with regards to an OEL (Occupational Exposure Limit). In addition to multiple illustrative graphs, it exposes five components.
- Goodness of fit with respect to the lognormal model (graphical evaluation).
- Descriptive statistics.
- Risk assessment based on an exceedance fraction.
- Risk assessment based on percentiles.
- Risk assessment based on the arithmetic mean.
Calculations are performed using a Bayesian model fit using a Monte Carlo Markov Chain (MCMC) engine. It assumes that the underlying exposure distribution is lognormal.
The underlying Bayesian models and data interpretation procedures are derived from best practices in industrial hygiene data interpretation techniques. They are thoroughly described in Expostats: A Bayesian Toolkit to Aid the Interpretation of Occupational Exposure Measurements (Annals of Work Exposures and Health, Volume 63, Issue 3, April 2019, Pages 267–279).
R version 4.4.0 is required to work on and serve Tool 1 locally. These
packages (and their transitive dependencies) are required.
- shiny
- bslib
- ggplot2
- ggimage
- htmltools
- rjags
- randtoolbox
- transltr
To serve Tool 1 locally, call
.run().run() is defined in .scripts/run.R.
Tool 1 is deployed to shinyapps.io.
To deploy a new version, call .pub().
# Deploy a beta version (useful for testing purposes).
# It is publicly accessible at https://lavoue.shinyapps.io/tool1-beta/.
.pub()
# Deploy an official (production) version.
# It is publicly accessible at https://lavoue.shinyapps.io/tool1/.
.pub("prod").pub() is defined in .scripts/publish.R.
Some environment variables are required to publish new releases with .pub().
They must be stored in a top-level .Renviron file as shown below. This file is
ignored by Git and rsconnect.
# .Renviron
RSCONNECT_ACCOUNT_NAME=<account>
RSCONNECT_ACCOUNT_TOKEN=<token>
RSCONNECT_ACCOUNT_SECRET=<secret>
The project is organized as a standard Shiny application as follows.
./
├── .local/
|   └── [temporary files ignored by Git scoped to each developer]
|
├── .scripts/
|   └── [scripts used while developing and ignored at runtime]
|
├── i18n/
|   └── [data used for internationalization (i18n) purposes]
|
├── man-roxygen/
|   └── [shared/common roxygen2 tags used more than once]
|
├── R/
|   └── [objects, constants, functions, etc.]
|
├── rsconnect/
|   └── [metadata on latest deployments]
|
├── scripts/
|   └── [official Expostats scripts and functions]
|
├── wwww/
│   ├── assets/
│   |   └── images/
|   |       └── [static images (any format)]
|   └── [static files served at runtime under root path]
|
└── [usual top-level source files]
Most objects are defined in .Rprofile, app.R, and in R/. For historical
reasons, all Expostats functions are stored in scripts/. They are explicitly
sourced at runtime by R/global.R. Other files are sourced automatically.
Static assets are stored in www/ and served under the root URL at runtime.
For example, file www/favicon.ico is served under /favicon.ico.
snake_case_with_lower_cases is used at all times.
CSS classes use dash-case-with-lower-cases for consistency with usual
best practices in web development. Each CSS class name must be prefixed
by app-.
For historical reasons (again), scripts stored in scripts/ do not reference
namespaces (packages). Consequently, some packages are explicitly attached to
the search path (see R/global.R). This is a bad practice from which Tool 1
is moving away.
# Good
transltr::language_source_get()
# Bad
language_source_get()We are actively looking for external collaborators who could help us with supporting more languages. If you are interested, please send an e-mail to [email protected].
Tool 1 relies on package transltr
to support multiple languages. Translations (and related metadata) are stored
in i18n/. To update its content, call
.find().find() is defined in .scripts/find-text.R.
Tool 1 further supports translation of ordinal numbers. Each supported
language requires an ordinal_rules_<lang>() function. For example, the
ordinal_rules_english() function implements grammar rules for English
ordinal numbers. See R/i18n.R for more information.
All files are required at runtime and read by transltr::translator_read()
to create global constant tr.
The _translator.yml file must never be modified manually. It should not
be shared with collaborators working on translations. Notably, developers may
consult it to locate translations in the source code.
Further <lang>.txt files contain actual translations. These files are shared
with collaborators working on translations and must be edited manually (using
any text editor). They always include basic instructions to follow at all times.
TRANSLATIONS.md contains further instructions for working with translation
files.
Tool 1 uses sprintf()-placeholders (conversion specifications beginning by
%) to dynamically insert HTML content into template strings. Tokens such as
%s, %i, and %% in the source text and translations must be left as is.
See R/html.R for more information.
Follow these steps to support a new language. Use the existing code as a template to follow.
- 
Implement a dedicated ordinal_rules_<lang>()function inR/i18n.R.
- 
Incorporate the function created at step (1) into ordinal_rules()inR/i18n.R.
- 
Add a new entry to formal argument other_lang_namesof.find()in.scripts/find-text.R. Follow instructions contained in the script to do so properly.
- 
Implement the required button and observer in ui_title()andserver_title()respectively. Follow instructions contained inR/ui-title.Rto do so (see subsection Languages).
- 
Call .find().
- 
Share the i18n/<lang>.txtfile created at step (5) with collaborator(s) in charge of translating Tool 1.- Never share the _translator.ymlfile. The latter is only useful to developers.
 
- Never share the 
- 
Import the i18n/<lang>.txtfile updated at step (6) back into the source code. Commit it.
You may repeat steps (5) to (7) whenever required. Notably, they must be
repeated whenever the source text is changed (whenever a call to translate()
is either added or updated).
Tool 1 uses as few custom CSS (Cascading Style Sheets) rules as possible. It
does so by maximizing usage of bslib features and predefined
Bootstrap 5
CSS classes. Any CSS rule not declared in file www/main.css is highly likely
to stem from Bootstrap 5.
This is a well-known and well-maintained framework that can be trivially
understood and leveraged.
It is worthwhile to note that it is not necessary to include
Bootstrap's assets
(in the <head> of Tool 1) because bslib already does that automatically.
We may work on these issues in a near future.
- 
Inputs lack robust validation mechanisms and may lead to undefined behavior in some cases. 
- 
Accessibility mechanisms (ARIA) are not implemented. 
- 
Margins, aspect ratios, font sizes, font families, dimensions, and colors of all plots are inconsistent and must be standardized. - This could be achieved with shiny::getCurrentOutputInfo()andbslib::bs_current_theme().
 
- This could be achieved with 
- 
There are currently no explicit Terms of Service and Privacy Policy. 
You may submit bugs, request features, and provide feedback by creating an issue on GitHub.
If you do not have a GitHub account, send an email to [email protected] and [email protected]. Please use the following standard subject line:
Expostats [Tool 1] [Bug|Feedback]: <short description>