Requiring Closure Namespaces
Yet another post on properly using the Closure Library from within ClojureScript. This time we’ll discuss how to require different namespaces from Closure and the edge-cases that may not be immediately intuitive.
Namespaces, Constructors, Constants
When requiring things from Closure you mostly deal with its namespaces. Most namespaces have functions defined in them, some also contain constructors or constants. Functions are camelCased. Constructors are Capitalized. Constants are ALL_CAPS. The line between namespaces and constructors gets a bit blurry sometimes as you’ll see shortly.
Let’s take goog.Timer
as an example. As per the previous paragraph you can infer that Timer
is a constructor. Just like in Clojure we use :import
to make constructors available:
(ns my.app
(:import [goog Timer]))
Now we may use the Timer
constructor as follows:
(def our-timer (Timer. interval))
Great. We have a timer. Now we’ll want to do something whenever it
“ticks”. The Timer
instance emits events which we can listen
to. Listening to events can be done with the function
goog.events.listen
. As you can see, this function is not part of any
class instance - it just exists in the goog.events
namespace.
To make the listen
function accessible you need to require the
namespace containing it. This is similar to how we require regular
ClojureScript namespaces:
(ns my.app
(:require [goog.events :as events])
(:import [goog Timer]))
We can refer to the function as events/listen
now. To listen to
specific kinds of events we need to pass an event type to this function. Many
Closure namespaces define constants that you can use to refer to
those event types. Internally they’re often just strings or numbers but
this level of indirection shields you from some otherwise breaking changes to
a namespace’s implementation.
Looking at the Timer
docs you can find a constant TICK
. Now we required the constructor
and are able to use that but the constructor itself does not allow us
to access other parts of the namespace. So let’s require the namespace.
(ns my.app
(:require [goog.events :as events]
[goog.Timer :as timer]) ; <-- new
(:import [goog Timer]))
(def our-timer (Timer. interval))
(events/listen our-timer timer/TICK (fn [e] (js/console.log e)))
Remember the blurry line mentioned earlier? We just required the goog.Timer
namespace
both as a constructor and as a namespace. While this example works
fine now, there are two more edge cases worth pointing out.
Deeper Property Access
Closure comes with a handy namespace for keyboard shortcuts, aptly named KeyboardShortcutHandler
.
As you can guess, KeyboardShortcutHandler
is a constructor that we can use via :import
.
Since it emits events, the namespace also provides an enum of events that we can use to listen for specific ones.
In contrast to the timer’s TICK
, this enumeration is “wrapped” in goog.ui.KeyBoardShortcutHandler.EventType
.
The EventType
property contains SHORTCUT_PREFIX
and SHORTCUT_TRIGGERED
. So far we’ve only imported the constructor.
At this point you might try this:
(:require [goog.ui.KeyBoardShortcutHandler.EventType :as event-types])
But that won’t work. The EventType
is not a namespace but an enum provided by
the KeyboardShortcutHandler
namespace. To access the enum you need to access it through the
namespace providing it. In the end this will look like this:
(:require [goog.ui.KeyBoardShortcutHandler :as shortcut])
(events/listen a-shortcut-handler shortcut/EventType.SHORTCUT_TRIGGERED ,,,)
Note how the slash always comes directly after the namespace alias.
goog.string.format
Last but not least another weird one. goog.string.format
is a namespace
that
seems to contain
a single function called format
. If you require the format namespace
however, it turns out to contain no function of that name:
(:require [goog.string.format :as format])
(format/format ,,,) ; TypeError: goog.string.format.format is not a function
Now in cases like this it often helps to look at the source code directly. Usually Closure Library code is very readable. The format function is defined as follows:
goog.string.format = function(formatString, var_args) {
As you can see it’s defined as a property of goog.string
, so we can
access it via goog.string/format
(or an alias you might have chosen
when requiring goog.string
). In that sense goog.string.format
is
not a real namespace but rather something you require for its side
effects — in this case the definition of another function in goog.string
.
I have no idea why they chose to split things up in that way.
¯\(ツ)/¯
For Reference
I scratched my head many times about one or the other aspect of this and usually ended up looking at old code. Next time I’ll look at the handy list below 🙂
- Require Google Closure namespaces just as you’d require ClojureScript namespaces
(:require [goog.events :as events])
- The base
goog
namespace is autmatically required as if you’d have[goog :as goog]
in your list of required namespaces.- This implies that you can refer to
goog.DEBUG
asgoog/DEBUG
. Never refer togoog
through the global Javascript namespace as injs/goog.DEBUG
. (CLJS-2023)
- This implies that you can refer to
- Require constructors using one of the two forms. In either case you may use
Timer.
to construct new objects.(:import [goog Timer])
(:import goog.Timer)
- There’s an outstanding ticket about imports with the same name shadowing each other.
- Only access non-constructor parts of a namespace through a namespace that has been
:require
d - Always use slash after the namespace alias, use dot for deeper property access.
- Requiring
goog.string.format
will define a functionformat
in thegoog.string
namespace.
Enjoy
For many of the things described here there are alternative ways to do them. We still build on Javascript after all. The ones I’ve chosen here are the ones that seem most idiomatic from a Clojurescript perspective.
Thanks to Paulus Esterhazy and António Monteiro for proof-reading this post and offering their suggestions.
If you feel like reading more about utilizing the Closure Library and compiler in ClojureScript I have a few more posts on those:
- Simple Debouncing in ClojureScript, showing how to build a simple debouncing mechanism with the facilities provided by the Closure Library.
- Parameterizing ClojureScript Builds, outlining ways to modify ClojureScript builds using the Closure compiler’s ability to customize constants at compile-time.
- Just-in-Time Script Loading, describing how to load 3rd party scripts like Stripe using React components and Closure’s script loader.