Experimental, more robust, and more user-friendly version of base R substitute
.
substitute2(expr, env)
Quoted expression having variables and call argument names substituted.
Unevaluated expression in which substitution has to take place.
List, or an environment that will be coerced to list, from which variables will be taken to inject into expr
.
For convenience function will turn any character elements of env
argument into symbols. In case if character is of length 2 or more, it will raise an error. It will also turn any list elements into list calls instead. Behaviour can be changed by wrapping env
into I
call. In such case any symbols must be explicitly created, for example using as.name
function. Alternatively it is possible to wrap particular elements of env
into I
call, then only those elements will retain their original class.
Comparing to base R substitute
, substitute2
function:
substitutes calls argument names as well
by default converts character elements of env
argument to symbols
by default converts list elements of env
argument to list calls
does not accept missing env
argument
evaluates elements of env
argument
substitute
, I
, call
, name
, eval
## base R substitute vs substitute2
substitute(list(var1 = var2), list(var1 = "c1", var2 = 5L))
substitute2(list(var1 = var2), list(var1 = "c1", var2 = 5L)) ## works also on names
substitute(var1, list(var1 = "c1"))
substitute2(var1, list(var1 = I("c1"))) ## enforce character with I
substitute(var1, list(var1 = as.name("c1")))
substitute2(var1, list(var1 = "c1")) ## turn character into symbol, for convenience
## mix symbols and characters using 'I' function, both lines will yield same result
substitute2(list(var1 = var2), list(var1 = "c1", var2 = I("some_character")))
substitute2(list(var1 = var2), I(list(var1 = as.name("c1"), var2 = "some_character")))
## list elements are enlist'ed into list calls
(cl1 = substitute(f(lst), list(lst = list(1L, 2L))))
(cl2 = substitute2(f(lst), I(list(lst = list(1L, 2L)))))
(cl3 = substitute2(f(lst), list(lst = I(list(1L, 2L)))))
(cl4 = substitute2(f(lst), list(lst = quote(list(1L, 2L)))))
(cl5 = substitute2(f(lst), list(lst = list(1L, 2L))))
cl1[[2L]] ## base R substitute with list element
cl2[[2L]] ## same
cl3[[2L]] ## same
cl4[[2L]] ## desired
cl5[[2L]] ## automatically
## character to name and list into list calls works recursively
(cl1 = substitute2(f(lst), list(lst = list(1L, list(2L)))))
(cl2 = substitute2(f(lst), I(list(lst = list(1L, list(2L)))))) ## unless I() used
last(cl1[[2L]]) ## enlisted recursively
last(cl2[[2L]]) ## AsIs
## using substitute2 from another function
f = function(expr, env) {
eval(substitute(
substitute2(.expr, env),
list(.expr = substitute(expr))
))
}
f(list(var1 = var2), list(var1 = "c1", var2 = 5L))
Run the code above in your browser using DataLab