Kitson P. Kelly avatar

A recovering 7 foot tall cactus

dojo/parser

Ok, what is becoming a bit of a trend, I am adding a third to my series on forthcoming features in Dojo core in 1.8. I have already covered dojo/promise and dojo/request, now I am going to cover changes to dojo/parser. This is something I am quite passionate about, since I had my hand directly involved in the features that were added, with lots of help from Bill Keese and Ben Hockey (and others).

With the full conversion to AMD in 1.7, there were likely a few features that made the declarative syntax a bit more challenging to use, versus the programmatic syntax. While the programmatic approach to Dojo is usually a better way of creating applications, the declarative way is very popular with developers, great for prototyping and a quick way to get started with Dojo, especially when using the Dijit package of widgets.

For those not familiar with programmatic, versus declarative syntax, here is a quick example of creating a Dijit button and making it log something to the console when clicked. First here is how we would do it programatically:

require(["dijit/form/Button"], function (Button) {
  var button = new Button({
    label: "Click Me!",
  }, "buttonNode");

  button.on("click", function () {
    console.log("I was clicked!");
  });
});

Now the same HTML snippet done declaratively:

<button type="button" id="buttonNode" data-dojo-type="dijit.form.Button">
  <span>Click Me!</span>
  <script type="dojo/on" data-dojo-event="click">
    console.log("I was clicked!");
  </script>
</button>

The declarative syntax is then converted by the dojo/parser into instantiated widgets, essentially converting your decorated HTML code into programmatic code. It does the heavy lifting for you, so you don’t have to keep managing the nodes in your document for all your widgets.

Now for the improvements in 1.8. The first was that you could only specify the class in declarative syntax using a global class name (the old way, dot notation, e.g. dijit.form.Button). With AMD and the deprecation of Dojo declaring classes in the global namespace, this was starting to become a problem. So now the dojo/parser accepts the Module ID (MID) as the type (e.g. dijit/form/Button). In fact this is now the preferred way of referring to a object type in declarative markup. For example, our button from above should be:

<button type="button" id="buttonNode" data-dojo-type="dijit/form/Button">
  <span>Click Me!</span>
  <script type="dojo/on" data-dojo-event="click">
    console.log("I was clicked!");
  </script>
</button>

The second added feature is to bring the declarative scripting fully capable with the introduction of dojo/aspect as the preferred way of modifying the behaviour of an instance in Dojo. dojo/aspect supports the semantic concepts of before, around and after which are part of Aspect Oriented Programming (AOP). Now the declarative scripting supports those as well, so you can do the following:

<div data-dojo-type="...">
  <script
    type="dojo/aspect"
    data-dojo-advice="after"
    data-dojo-method="method1"
    data-dojo-args="e"
  >
    console.log("I ran after!");
  </script>
  <script
    type="dojo/aspect"
    data-dojo-advice="around"
    data-dojo-method="method2"
    data-dojo-args="origFn"
  >
    return function () { // Have to act as a function factory
      console.log("I ran before!");
      origFn.call(this); // You have to call the original function
      console.log("I ran after!");
    };
  </script>
  <script
    type="dojo/aspect"
    data-dojo-advice="before"
    data-dojo-method="method3"
    data-dojo-args="i"
  >
    console.log("I ran before!");
    i++; // Modifying argument
    return [i]; // Returning modified arguments to be used with original function
  </script>
</div>

Now for the “cool” feature, but which could easily cause quite unintended consequences, is something called auto-require. This caused a lot of debate within the Dojo developer community, mostly because you don’t have to be explicit about your requirements with your declarative scripting. Basically if the dojo/parser encounters a MID that isn’t loaded, it will attempt to load it for you. It makes things easier, but it can mask problems and may cause a lot of negative impact on performance when using built layers that may not contain all the right modules. At the end of the day though we thought it was useful enough to be included. Without it, you would have had to require all your modules used in your declarative markup before invoking the parser, which would have looked something like this for our button:

require(
  ["dojo/ready", "dojo/parser", "dijit/form/Button"],
  function (ready, parser) {
    ready(function () {
      parser.parse();
    });
  },
);

:information_source: It is still better, since 1.7 to manually invoke the parser.parse() instead of using parseOnLoad in dojoConfig.

Another new feature which fits well in the world of AMD is a declarative require. This allows you to require in modules without having to do it within a JavaScript code block in your code. What it essentially does, is allow you to load modules and map them into the global namespace. A declarative require is a <script> block that contains a JavaScript object that defines the mapping of the modules and would look like this:

<script type="dojo/require">
  "app.on": "dojo/on",
  myModule: "package/myModule"
</script>

Because of the auto-require and declarative require use Dojo’s loader (via require()) and require() operates in an async fashion, it has meant that dojo/parser now operates in a async fashion. Historically the .parse() function returned an array of instantiated widgets. For backwards compatibility reasons .parse() still returns an array, but it also behaves like a promise too. When the parser does not need to use auto-require or declarative require or Dojo loader is running in sync mode, this array will be populated with the instantiated objects. When it is running in an async fashion, the array will be empty but the promise will be resolved with an array of objects.

For new development it is best to ensure you treat parser.parse() as if it is always running in an async fashion. So if you need access to the instantiated objects or if you need to do something once the .parse() is finished, you should do something like this:

require(["dojo/parser", "dojo/ready"], function (parser, ready) {
  ready(function () {
    parser.parse().then(function (instances) {
      // do something
    });
  });
});

One piece that I am still working as I write this is to get the Dojo builder to be able to analyse your HTML resources for a build, understand the dependencies in the marked up code and allow you to build that into your built Dojo or a layer. Hopefully this will help people avoid some of the issues that could occur if they just used features like auto-require without thinking about the consequences.

There is a lot of “good” stuff in the dojo/parser in 1.8 that makes it more “modern” like other parts of Dojo. Hopefully for those who use it, these features will help you take advantage of a modern Dojo and feel “left behind”.