-
-
Save alandipert/2346460 to your computer and use it in GitHub Desktop.
(extend-type js/RegExp | |
cljs.core/IFn | |
(-invoke ([this s] (re-matches this s)))) | |
(#"foo.*" "foobar") ;=> "foobar" | |
(#"zoo.*" "foobar") ;=> nil | |
(filter #".*foo.*" ["foobar" "goobar" "foobaz"]) ;=> ("foobar" "foobaz") |
Appropos: proposal for something like this in JS and library
https://mail.mozilla.org/pipermail/es-discuss/2012-March/021603.html
https://github.com/Gozala/protocol
It's a little known fact that ClojureScript does this (extends) already with strings to support keyword and symbol lookup. :-)
@fogus Interesting! Could you point me to the source for this so I can see how it's done? I ack'd the ClojureScript source for extend-type
and extend
but didn't find the magic. :)
@mjijackson the behavior @fogus mentions is incidental, and is a result of the way :keywords
are implemented atop JavaScript strings. That said, I would classify this behavior as seekrit and not sweet because it depends on implementation detail.
But then, maybe seekrit things are sweet?
This Old Gist™
Despite the fact that this is seekrit I still point it out to clojure journeyfolk because I think it teaches deep Clojure by way of radical ClojureScript in one beautiful motion.
Nowadays, no longer need to hide seekrits. Keywords and symbols are actually types that support IFn
:
https://github.com/clojure/clojurescript/blob/r1.7.228/src/main/cljs/cljs/core.cljs#L2941-L2945
https://github.com/clojure/clojurescript/blob/r1.7.228/src/main/cljs/cljs/core.cljs#L945-L949
@mfikes I can understand keywords, but why are symbols functions of collections? That seems like a juicy one to leave open for users.
Edit
Nevermind, answered my own question. Clojure does the same thing, which must have been the rationale. Fortunately regexes and many other atoms are available for IFn extension 😈
By way of explanation for anyone else who stumbles upon this gist, the
IFn
protocol is used for adding the ability to invoke an object as a function. In this example, instances of JavaScript's built-inRegExp
constructor are made callable. This means you can use them in the same place you would normally use a function. As implemented, the result of invoking aRegExp
object is the return value of the re-matches function using that regular expression and the given argument.For those who understand JavaScript better than ClojureScript, this would be equivalent to the following JavaScript (which does not actually work, hence the sweet-ness):